在'ssh'的popen()上调用fgets()正在刷新调用进程的stdin的开头(ptty问题)

时间:2009-09-02 03:03:24

标签: ssh stdin popen fgets pty

我现在已将其缩减到最小的测试用例。到目前为止,我已经能够确定这是与伪终端相关的问题,这些伪终端与ssh的管道有关。将'-t -t'添加到ssh调用改进了一些东西,现在,它需要再次调用fgets()来引发问题。我怀疑ssh命令的stderr输出以某种方式解决了这个问题,现在我已经将sder代码中的stderr重定向到stdout来执行。我想知道“tcgetattr:Invalid argument”错误是否是问题的一部分,但我不确定如何摆脱它。它似乎来自-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 ????

1 个答案:

答案 0 :(得分:1)

好的,我终于有了这个工作。秘诀是提供/ dev / null作为我的ssh命令的输入,如上面的测试用例所示:

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

然而,当代码正常工作时,我收到一条令人讨厌的消息,显然我可以忽略我的目的(虽然我想让消息消失):

tcgetattr: Inappropriate ioctl for device