通过把手向下管道

时间:2017-11-13 18:19:41

标签: node.js macos stdin pid file-descriptor

说我有

node foo.js | node bar.js

有没有办法将foo&#s; stdin上的句柄传递给bar.js?

我有一个罕见的案例,我想在管道中向后沟通。

至少我知道我可以发送node bar.js node foo.js的pid。鉴于pid,在* nix上,我应该可以使用以下命令写入foo的stdin:

/proc/<pid>/fd/0

但有没有办法在MacOS上做同样的事情?

1 个答案:

答案 0 :(得分:6)

所以有不同的方法。

方法1 - IOCTL

这是受到

的启发

https://stackoverflow.com/a/36522620/2830850

因此,您创建了包含以下内容的writevt.c文件

/*
 * Mostly ripped off of console-tools' writevt.c
 */

#include <stdio.h>
#include <fcntl.h>
#include <termios.h>
#include <sys/ioctl.h>
#include <unistd.h>

char *progname;

static int usage() {
    printf("Usage: %s ttydev text\n", progname);
    return 2;
}

int main(int argc, char **argv) {
    int fd, argi;
    char *term = NULL;
    char *text = NULL;

    progname = argv[0];

    argi = 1;

    if (argi < argc)
        term = argv[argi++];
    else {
        fprintf(stderr, "%s: no tty specified\n", progname);
        return usage();
    }

    if (argi < argc)
        text = argv[argi++];
    else {
        fprintf(stderr, "%s: no text specified\n", progname);
        return usage();
    }

    if (argi != argc) {
        fprintf(stderr, "%s: too many arguments\n", progname);
        return usage();
    }

    fd = open(term, O_RDONLY);
    if (fd < 0) {
        perror(term);
        fprintf(stderr, "%s: could not open tty\n", progname);
        return 1;
    }

    while (*text) {
        if (ioctl(fd, TIOCSTI, text)) {
            perror("ioctl");
            return 1;
        }
        text++;
    }

    return 0;
}

使用下面的

编译它
gcc -o writevt writevt.c

然后将root权限添加到同一

sudo chown root:wheel writevt
sudo chmod 4755 writevt

现在我使用以下代码

创建了一个简单的foo.js
var stdin = process.openStdin();

stdin.addListener("data", function(d) {
    console.log(process.env.NAME + " entered: [" +
        d.toString().trim() + "]");
});

在终端中首先运行tty命令

$ tty
/dev/ttys019

现在运行如下代码

NAME=A node foo.js  | NAME=B node foo.js

现在从另一个终端运行以下命令

./writevt /dev/ttys019 "FROM external command^M"

^M此处为Mac上的CTRL+V + CTRL+ENTER

Content

gif可以看到输入到达A的stdin然后A打印在stdout上,然后由B接收。所以如果我修改下面的代码

var stdin = process.openStdin();

stdin.addListener("data", function(d) {
    console.log(process.env.NAME + " entered: [" +
        d.toString().trim() + "]");
});

if (process.env.NAME === "B") {
    setInterval(function() {
        require('child_process').exec(`./writevt /dev/ttys019 "Hello from B?
"`)
    }, 1000)
}

注1 :在上面的代码中使用Vim添加了^M

注意2 :TTY位置已在此硬编码,但您可以通过运行

将其传递给环境变量
export TTY=`tty`

然后在代码中使用process.env.TTY。更新的结果是

Working

方法2 - FIFO文件

在这种方法中,您首先制作一个fifo文件

$ mkfifo nodebridge

现在您更改下面的代码

var stdin = process.openStdin();
var fs = require("fs")
stdin.addListener("data", function(d) {
    console.log(process.env.NAME + " entered: [" +
        d.toString().trim() + "]");
});

if (process.env.NAME === "B") {
    setInterval( () => {
        require('child_process').exec('printf "Hello from B?\\n" > nodebridge')
    }, 1000);
}

然后运行如下命令

NAME=A node foo.js < nodebridge | NAME=B node foo.js

NodeBridge