Вызов fgets() для popen() из 'ssh' сбрасывает начало стандартного ввода вызывающего процесса (проблема с ptty)

Теперь я сократил это до минимального тестового случая. До сих пор я смог определить, что это проблема, связанная с псевдотерминалом, который возникает с каналом ssh. Добавление '-t -t' к вызову ssh улучшило ситуацию, поскольку теперь для возникновения проблемы требуется второй вызов fgets(). Я подозреваю, что вывод stderr команды ssh каким-то образом влияет на проблему, а пока я перенаправил stderr на stdout в коде ssh для выполнения. Мне интересно, является ли ошибка «tcgetattr: Invalid arguments» частью проблемы, но я не знаю, как от нее избавиться. Кажется, это происходит от присутствия -t -t. Я считаю, что -t -t движется в правильном направлении, но мне нужно как-то настроить псевдотерминал для stderr, и, возможно, тест будет работать правильно?

Makefile:

test:
    gcc -g -DBUILD_MACHINE='"$(shell hostname)"' -c -o test.o test.c
    gcc -g -o test test.o

.PHONY: clean
clean:
    rm -rf test.o test

Исходный файл test.c:

#include <unistd.h>
#include <string.h>
#include <stdio.h>

int
main(int argc, char *argv[])
{
  const unsigned int bufSize = 32;
  char buf1[bufSize];
  char buf2[bufSize];
  int ssh = argv[1][0] == 'y';
  const char *cmd = ssh ? "ssh -t -t " BUILD_MACHINE " \"ls\" 2>&1" : "ls";

  FILE *fPtr = popen(cmd, "r");

  if (fPtr == NULL) {
    fprintf(stderr,"Unable to spawn command.\n");
        perror("popen(3)");
        exit(1);
  }
  printf("Command: %s\n", cmd);
  if (feof(fPtr) == 0 && fgets(buf2, bufSize, fPtr) != NULL) {
    printf("First result: %s\n", buf2);
    if (feof(fPtr) == 0 && fgets(buf2, bufSize, fPtr) != NULL) {
      printf("Second result: %s\n", buf2);
      int nRead = read(fileno(stdin), buf1, bufSize);

      if (nRead == 0) {
        printf("???? popen() of ssh consumed the beginning of stdin ????\n");
      } else if (nRead > 0) {
        if (strncmp("The quick brown fox jumped", buf1, 26) != 0) {
          printf("??? Failed ???\n");
        } else {
          printf("!!!!!!!   Without ssh popen() did not consume stdin   !!!!!!!\n");
        }
      }
    }
  }
}

Это показывает, как он работает попутно:

> echo "The quick brown fox jumped" | ./test n
Command: ls
First result: ARCH.linux_26_i86

Second result: Makefile

!!!!!!!   Without ssh popen() did not consume stdin   !!!!!!!

Это показывает, что он работает с ошибкой:

> echo "The quick brown fox jumped" | ./test y
Command: ssh -t -t hostname "ls" 2>&1
First result: tcgetattr: Invalid argument

Second result: %backup%~              gmon.out

???? popen() of ssh consumed the beginning of stdin ????

person Community    schedule 02.09.2009    source источник


Ответы (1)


Хорошо, у меня наконец-то это работает. Секрет заключался в том, чтобы указать /dev/null в качестве входных данных для моей команды ssh, как следует из приведенного выше тестового примера:

      const char *cmd
        = ssh ? "ssh -t -t " BUILD_MACHINE " \"ls\" 2>&1 < /dev/null" : "ls";

Однако, хотя код работает правильно, я получаю неприятное сообщение, которое, по-видимому, я могу игнорировать для своих целей (хотя я бы хотел, чтобы сообщение исчезло):

tcgetattr: Inappropriate ioctl for device
person Community    schedule 19.09.2009
comment
попробуйте направить вывод в /dev/null, а не ОТ него. - person GaZ; 04.12.2012