如何在Firefox附加组件中发送子进程,如Chrome本机消息传递

Kur*_*urt 7 javascript python firefox-addon firefox-addon-sdk chrome-native-messaging

我正在尝试使用Firefox的附加SDK 模拟Chrome的本机消息功能.具体来说,我正在使用child_process模块emit方法与python子进程通信.

我能够成功地向子进程发送消息,但是我无法将消息发送回附加组件.Chrome的本机消息传递功能使用stdin/stdout.两个方向上每个消息的前4个字节表示后续消息的字节大小,因此接收方知道要读取多少.这是我到目前为止所拥有的:

子进程的附加组件

var utf8 = new TextEncoder("utf-8").encode(message);
var latin = new TextDecoder("latin1").decode(utf8);

emit(childProcess.stdin, "data", new TextDecoder("latin1").decode(new Uint32Array([utf8.length])));
emit(childProcess.stdin, "data", latin);
emit(childProcess.stdin, "end");
Run Code Online (Sandbox Code Playgroud)

来自附加组件的子进程(Python)

text_length_bytes = sys.stdin.read(4)
text_length = struct.unpack('i', text_length_bytes)[0]
text = sys.stdin.read(text_length).decode('utf-8')
Run Code Online (Sandbox Code Playgroud)

加载项的子进程

sys.stdout.write(struct.pack('I', len(message)))
sys.stdout.write(message)
sys.stdout.flush()
Run Code Online (Sandbox Code Playgroud)

子进程的附加组件

这是我在努力的地方.当长度小于255时,我可以使用它.例如,如果长度为55,则可以:

childProcess.stdout.on('data', (data) => { // data is '7' (55 UTF-8 encoded)
    var utf8Encoded = new TextEncoder("utf-8).encode(data);
    console.log(utf8Encoded[0]); // 55
}
Run Code Online (Sandbox Code Playgroud)

但是,就像我说的那样,它并不适用于所有数字.我确定我必须对TypedArrays做些什么,但我很难把所有东西放在一起.

hum*_*oid 4

这里的问题是 Firefox 默认尝试将 stdout 读取为 UTF-8 流。由于 UTF-8 不使用完整的第一个字节,因此您会得到损坏的字符,例如 255。解决方案是告诉 Firefox 以二进制编码读取,这意味着您稍后必须手动解析实际的消息内容。

var childProcess = spawn("mybin", [ '-a' ], { encoding: null });
Run Code Online (Sandbox Code Playgroud)

然后你的听众会像这样工作

var decoder = new TextDecoder("utf-8");
var readIncoming = (data) => {
    // read the first four bytes, which indicate the size of the following message
    var size = (new Uint32Array(data.subarray(0, 4).buffer))[0];
    //TODO: handle size > data.byteLength - 4
    // read the message
    var message = decoder.decode(data.subarray(4, size));
    //TODO: do stuff with message
    // Read the next message if there are more bytes.
    if(data.byteLength > 4 + size)
        readIncoming(data.subarray(4 + size));
};
childProcess.stdout.on('data', (data) => {
    // convert the data string to a byte array
    // The bytes got converted by char code, see https://dxr.mozilla.org/mozilla-central/source/addon-sdk/source/lib/sdk/system/child_process/subprocess.js#357
    var bytes = Uint8Array.from(data, (c) => c.charCodeAt(0));
    readIncoming(bytes);
});
Run Code Online (Sandbox Code Playgroud)