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

时间:2016-04-16 01:42:16

标签: javascript python firefox-addon firefox-addon-sdk chrome-native-messaging

我正在尝试使用Firefox的附加SDK模仿Chrome的native messaging feature。具体来说,我正在使用child_process moduleemit method与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");
来自附加组件的

子进程(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')

加载项的子进程

sys.stdout.write(struct.pack('I', len(message)))
sys.stdout.write(message)
sys.stdout.flush()

子进程的附加组件

这是我在努力的地方。当长度小于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
}

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

2 个答案:

答案 0 :(得分:4)

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

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

然后你的听众会像

一样工作
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);
});

答案 1 :(得分:0)

这可能与此问题类似: Chrome native messaging doesn't accept messages of certain sizes (Windows)

  

仅限Windows:确保程序的I / O模式设置为O_BINARY。默认情况下,I / O模式为O_TEXT,这会破坏消息格式,因为换行符(\ n = 0A)将替换为Windows样式的行结尾(\ r \ n = 0D 0A)。可以使用__setmode设置I / O模式。