如何修复 Chrome 浏览器扩展中的 Native Host Has Exited 错误 -- Native Messaging

How to fix Native Host Has Exited error in Chrome browser extension -- Native Messaging

如何修复 Mac 的 Chrome 浏览器扩展中的 "Native host has exited" 错误? Native Messaging 上的文档在底部特别提到了这个错误,并提出了一个可能的原因,但它的描述相当笼统 ("This is most likely initiated from your native messaging host"),目前还不清楚如何尝试解决这个问题。

终端中的日志输出(如果您从 Mac 终端启动 Chrome,您可以查看日志信息)让我相信本机主机应用程序本身的进程永远不会启动成功。我不断收到:

LaunchProcess: failed to execvp:
/Users/mycomputer1/Documents/myhost.app

但是,将 "Native host has exited" 打印到后台 javascript 页面的控制台的实际代码是:

function onDisconnected() {
  console.log("Inside onDisconnected(): " + chrome.runtime.lastError.message);
  port = null;
}

onDisconnected() 侦听器实际上是调用 connectNative() 的结果,我已根据文档在此处完成此操作:

chrome.runtime.onMessageExternal.addListener(
function(request, sender, sendResponse) {
//var imgdata = JSON.stringify(request.imgdata);
//process it somehow here

    port = chrome.runtime.connectNative("com.allinlearning.ailhost");

    if (port)
    {
       console.log("connectNative() returned a non-null port");
    }
});

我知道 connectNative() 在某种程度上是成功的,因为它 returns 是一个有效的端口。但是,connectNative() 继续尝试定位本机主机应用程序并尝试启动它。文档表明主机应用程序是罪魁祸首,以某种方式启动了 Chrome 和本机应用程序之间的 "pipe" 中断,尽管我的主机应用程序根本没有实现与 stdout 或 stdin 的通信。此外,Chrome 日志消息 ("LaunchProcess: failed to execvp:") 似乎表明我的 Mac 应用程序甚至从未成功启动。

还有更多:我在网上找到的一些源代码(也许过时了?source file)准确地揭示了这个 "Native host has exited" 消息的来源,并且检查表明为什么 Google Chrome 的代码正在关闭。在 HandleReadResult() 中,调用 Close() 来提供消息,然后关闭事物以响应返回 0 的 Posix read()。但是在广阔的体育世界中,我做错了什么(或没有做错)在我的 Mac 应用程序中做任何事情)导致读取失败?我的 Mac 应用程序没有向标准输出写入任何内容,那么不写入任何内容是如何导致读取错误的呢?还是所有这些都误导了我,而 execvp 未能启动我的主机应用程序是更深层的原因?如果是这样,为什么?这是一个简单的准系统 Mac 应用程序。

您的 "myhost.app" 不是二进制文件,而是一个目录(如您的 u+w,go-w,a+rX 权限中的 X 所示)。

您有两个选择:

  1. 不要为您的命令行程序创建一个 App bundle,而是一个普通的命令行应用程序(Xcode 中的文件 > 新建 > 项目 > 命令行工具(tutorial)).

  2. 指定应用包内二进制文件的路径,而不是包本身。您可以通过右键单击 .app 文件并选择 "Show package contents" 来列出应用程序包中的所有文件。或者 运行 从终端(控制台)执行以下命令以列出您的捆绑包中的所有文件:

    find /Users/mycomputer1/Documents/myhost.app -type f
    

您的本机应用程序和 chrome 通过控制台进行通信,该控制台在大多数情况下是隐藏的。如果您的应用程序使用控制台的目的不仅仅是与 chrome 通信(例如执行系统命令),chrome 将捕获它并将其解释为来自您的应用程序的错误响应并关闭它们之间的连接。

因此建议您使用以下解决方法之一:

  • 除了与 chrome
  • 的通信之外,不要将(同一)控制台用于 运行 任何其他事情
  • 缓冲您的控制台输出而不是将其发送到标准输出

还有一件事要注意,在某些语言中(例如 php),某些命令(例如 preg_match)似乎没有任何输出,因为它们根据参数操作数据,但是他们做 return 结果(例如 true/false),如果你不抑制那个结果(php 中的@function())或将它保存到一个变量,那么它会转到 stdout并导致 chrome 断开连接。

抱歉,如果与问题不直接相关,但我希望这对某些人有所帮助,因为这个问题会出现在搜索结果中。就我自己而言,我在这个问题上苦苦挣扎了好几个小时。

发生在我​​身上 Linux (RHEL 7.6)。

要找到根本原因,您需要从控制台(命令行)运行 chrome,然后打开 gnome 扩展页面:

$ /opt/google/chrome/chrome "https://extensions.gnome.org"

然后查看控制台,是否有任何错误。

对我来说这是 'chrome-gnome-shell' 错误,因为缺少名为 'gi':

的 python 模块
Traceback (most recent call last):
Traceback (most recent call last):
  File "/usr/bin/chrome-gnome-shell", line 16, in <module>
    from gi.repository import GLib, Gio
ModuleNotFoundError: No module named 'gi'

要修复它,我必须安装 PyGObject:

$ sudo yum install python36-gobject.x86_64
# or directly with pip: pip3 install PyGObject

安装完成后,返回 https://extensions.gnome.org(可能需要关闭并重新打开 Chrome)- 现在没有显示任何错误,并且运行良好。