2016-10-16 8 views
13

私はstdinから入力を読み込み、入力を操作し、出力をstdoutに書き出すプログラムを書いています。しかし、多くのプログラムは、stdinが端末かパイプか(isattyのような関数を呼び出すことによって)チェックし、出力を別々に生成します。私のプログラムはどのようにしてTTYのふりをしますか?TTYのふりをするプログラムを書く

解決策は、LinuxとmacOSの両方で動作するはずです。スタンドアロンのバイナリを生成するプログラミング言語は使用できますが、Goが優先されます。

私はプログラミングの質問をしていますが、ツールを求めていません。だから、scriptunbufferのようなものは私が探しているものではありません。

+0

あそこ興味深いアプローチがあります。http://unix.stackexchange.com/a/249764/65275 – coredump

+0

おかげではなく、すべてのこれらのソリューションは、特定の外部に依存プログラム。私は、クライアントシステムに特定のものをインストールさせることなく動作するスタンドアローンプログラムを書く必要があります。 – Ethan

+3

[このUnix&Linux Stack Exchangeの質問](http://unix.stackexchange.com/q/21147/12106)がより適切かもしれません。あなたが本当に検索したいという言葉は* pseudo tty *(または* pty *)です。 –

答えて

1

以下は、ptyでコマンドを実行して出力をキャプチャするための完全な動作コードです。 (未できるだけ多くの行あなたが思っているかもしれませんが。)

#include <signal.h> 
#include <stdlib.h> 
#include <sysexits.h> 
#include <unistd.h> 
#include <util.h> 

pid_t child = 0; 

void sighandler(int signum) { 
    if (child > 0) { 
    killpg(child, signum); 
    exit(signum); 
    } 
} 

// Run a command in a pty. 
// Usage: /path/to/this/binary command to run 
int main(int argc, char *argv[]) { 
    if (argc < 2) { 
    return EX_USAGE; 
    } 

    int master; 
    child = forkpty(&master, NULL, NULL, NULL); 

    if (child == -1) { 
    perror("failed to fork pty"); 
    return EX_OSERR; 
    } 

    if (child == 0) { 
    // we're in the child process, so replace it with the command 
    execvp(argv[1], argv + 1); 
    perror("failed to execute command"); 
    return EX_OSERR; 
    } 

    // trap kill signals and forward them to child process 
    signal(SIGHUP, sighandler); 
    signal(SIGINT, sighandler); 
    signal(SIGTERM, sighandler); 

    const int buf_size = 1024; 
    char buf[buf_size]; 
    fd_set fds; 
    ssize_t bytes_read; 

    // forward the output continuously 
    while (1) { 
    FD_ZERO(&fds); 
    FD_SET(master, &fds); 

    if (select(master + 1, &fds, NULL, NULL, NULL) > 0 && FD_ISSET(master, &fds)) { 
     bytes_read = read(master, buf, buf_size); 
     if (bytes_read <= 0) { 
     return EXIT_SUCCESS; 
     } 

     if (write(STDOUT_FILENO, buf, bytes_read) != bytes_read) { 
     perror("failed to write to stdout"); 
     return EX_OSERR; 
     } 
    } 
    } 
}