Chrome extension与Native app通信

  • 内容概要
    一、了解chrome扩展与本地应用的通信协议
    二、实现简单的chrome扩展
    三、实现简单的本地的python应用
    四、绑定chrome扩展与Python应用
    五、应用间通信
    六、总结

  • 前言
    是什么?互为独立的应用通过既定规则进行通信
    为什么?在已有工具上进行功能的扩展,满足多方面的需求
    怎么样?First,了解protocol,then,创建crx和native app,at last, communicate.

一、Native Messaging Protocol

chrome规定了chrome-extensions与本地应用通信的协议:
简言之,首先需要将local application与chrome-extension绑定,然后通过标准的输入输出流来交换数据,数据的格式约定为header+body,header=sizeof(body), body为JSON格式(UTF8编码),本地应用最多可发送1M-bit而chrome-extensions最多可发送4G-bit。

python_app.png

二、Create a simple Chrome extension

simple_crx.png
  • manifest.json

Chrome extension与Native appliaiton信息交互需要获取权限nativeMessaging

{
  "name": "Native sendMessage Sample",
  "description": "Native sendMessage",
  "version": "0.1",
  "permissions": ["nativeMessaging", "*://*/*"],
  "background": {
    "scripts": ["simple.js"]
  },
  "browser_action": {},
  "manifest_version": 2
}
  • background.js

当点击浏览器扩展时向本地应用com.stone发送数据{text: "Hello, stonejianbu."},发送数据使用接口chrome.runtime.sendNativeMessage(reg_local_app,send-object,callback),注意chrome-extension作为发送端时不需要指明header(32-bit),chrome extension内部已作处理。

chrome.browserAction.onClicked.addListener(function () {
    chrome.runtime.sendNativeMessage("com.stone", {text: "Hello, stonejianbu."}, function (response) {
        console.log("Received " + response.response);
    });
});

三、Create a Python Application

native_app.png
  • Python应用程序源码
# crx_python.py
"""
chrome extension messaging with local application.
"""
import sys
import struct
import json
import logging


# 主函数入口
def main():
    # 1.对windows平台的行结束符处理:\r\n--->\n
    if sys.platform == "win32":
        import os, msvcrt
        msvcrt.setmode(sys.stdin.fileno(), os.O_BINARY)
        msvcrt.setmode(sys.stdout.fileno(), os.O_BINARY)

    # 2.日志文件处理对象
    logging.basicConfig(filename="ctl.log",
                        level=logging.INFO,
                        format="%(levelname)s::%(asctime)s-%(module)s-%(lineno)d: %(message)s",
                        datefmt="%Y-%d-%m %H:%M:%S")

    # 3.与chrome extension进行通信
    try:
        ctl = CrxToLocal()
        ctl.read_message()
        logging.info("3.1.正在接收数据...")
        ctl.handle_message()
        logging.info("3.2.正在处理数据...")
        ctl.send_message()
        logging.info("3.3.正在返回数据...")
    except Exception as e:
        logging.error(e)


# Chrome extension to local application:与chrome-extension进行通信的class
class CrxToLocal:
    def __init__(self):
        self._input_body = None
        self._output_body = None

    def read_message(self):
        # 读取标准输入流中的msg-header(first 4 bytes).
        text_length_bytes = sys.stdin.buffer.read(4)

        # Unpack message length as 4 byte integer, tuple = struct.unpack(fmt, buffer).
        text_length = struct.unpack("I", text_length_bytes)[0]

        # 读取标准输入流中的msg-body bytes
        self._input_body = sys.stdin.buffer.read(text_length)

    def handle_message(self):
        """更多处理对输入输出数据的操作..."""
        # 将chrome extension发送过来的数据保存到本地
        with open("./local_file.json", "a", encoding="utf-8") as f:
            json.dump(json.loads(self._input_body, encoding="utf-8"), f, ensure_ascii=False, indent=2)

        # _output_body的数据格式要求为JSON utf-8 bytes
        self._output_body = json.dumps({"response": "Nice to meet you."}).encode("utf-8")

    def send_message(self):
        # 将msg-header写到标准输出流, I表示int类型占4个字节,相关内容查看文档介绍python-doc/library/struct
        sys.stdout.buffer.write(struct.pack("I", len(self._output_body)))
        # 将msg-body写到标准输出流
        sys.stdout.buffer.write(self._output_body)
        sys.stdout.flush()


if __name__ == "__main__":
    main()

四、Bind chrome-extension with native application

windows下,通过给注册表添加键值以绑定chrome-extension与本地应用的关系:
1.首先创建一个native_manifest.json文件
2.添加Native_applicaiton和chrome-extension的描述信息
3.添加到注册表

  • native_manifest.json

1.path指定本地应用的可执行程序路径,此处的stone.bat内部执行了python应用
2.allowed_origins/chrome-extension指定与本地应用绑定的扩展的ID
3.name指定本地应用名称,与注册表注册的键对应,native_manifest.json文件将作为注册表中的值

{
  "name": "com.stone",
  "description": "My Application",
  "path": "C:\\Users\\Administrator\\Desktop\\native_app\\stone.bat",
  "type": "stdio",
  "allowed_origins": [
    "chrome-extension://mnmmaemegkeggpgimaedjkjlmncljibh/"
  ]
}
  • 将com.stone: native_manifest.json键值添加到windows注册表

将以下内容写入*.reg文件中,点击运行即注册成功(注意实际路径的修改)

Windows Registry Editor Version 5.00
[HKEY_CURRENT_USER\Software\Google\Chrome\NativeMessagingHosts\com.stone]
@="C:\\Users\\Administrator\\Desktop\\native_app\\native_manifest.json"

五、Communication

flow.png
    1. chrome-extension向Native app发送信息并等待响应
chrome.browserAction.onClicked.addListener(function () {
    chrome.runtime.sendNativeMessage("com.stone", {text: "Hello, stonejianbu."}, function (response) {
        console.log("Received " + response.response);
    });
});
    1. Native app接受chrome-extension发送的信息
    def read_message(self):
        # 读取标准输入流中的msg-header(first 4 bytes).
        text_length_bytes = sys.stdin.buffer.read(4)

        # Unpack message length as 4 byte integer, tuple = struct.unpack(fmt, buffer).
        text_length = struct.unpack("I", text_length_bytes)[0]

        # 读取标准输入流中的msg-body bytes
        self._input_body = sys.stdin.buffer.read(text_length)
  • 3.Native app向chrome-extension发送信息
    def send_message(self):
        # 将msg-header写到标准输出流, I表示int类型占4个字节,相关内容查看文档介绍python-doc/library/struct
        sys.stdout.buffer.write(struct.pack("I", len(self._output_body)))
        # 将msg-body写到标准输出流
        sys.stdout.buffer.write(self._output_body)
        sys.stdout.flush()
    1. chrome-extension等待接收Native app发送的数据

首先需要将chrome-extension和Native app之间建立长连接,创建chrome.runtime.Port object,注意在上面的应用中使用的是单一连接而未创建Port

var port = chrome.runtime.connectNative("com.stone");
port.onMessage.addListener(function(msg){
    console.log("Received"+msg);
});
port.onDisconnect.addListener(function(){
    console.log("Disconnected.");
});
port.postMessage({greet: "Hello, stonejianbu."});

六、总结

本文描述的仅是Chrome-extension与Native app通信流程及其简单使用,在此基础上,我们可以做功能上扩展如:
1.给chrome-extension添加pop.html由用户输入来控制和本地应用进行通信
2.给Native app添加GUI界面以控制多样化的通信

为了实现与本地应用或者远程服务器进行通信,我们可以使用更为通用的协议如http、ftp等等。如,首先建立本地的http服务器,然后管理后台background.js与服务器进行交互,示例代码如下:

"""manifest.json
{
  "name": "communication",
  "version": "1.0",
  "manifest_version": 2,
  "browser_action": {},
  "permissions": ["http://127.0.0.1/*"],
  "background": {
    "scripts": ["background.js"]
  }
}
"""
-------------------------------------------------------------------------------------------------
"""background.js
// 当点击扩展图标时触发
chrome.browserAction.onClicked.addListener(function () {
    console.log("hello world.");
    // 发送ajax post请求
    var xhttp = new XMLHttpRequest();
    xhttp.onreadystatechange = function () {
        if (this.readyState === 4 && this.status === 200) {
            console.log(this.responseText);
            console.log("成功发送到服务器...");
        } else {
            console.log("nothing......")
        }
    };
    xhttp.open("POST", "http://127.0.0.1:5000/index", true);
    xhttp.setRequestHeader("Content-type", "application/x-www-form-urlencoded"); // post请求时添加
    xhttp.send("hello=stone_world&name=stone");
});
"""

JavaScriptchrome-extension、python的structlogging模块的使用等请参考相关官方文档。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容