当我得到以下错误:

events.js:72
        throw er; // Unhandled 'error' event
              ^
Error: spawn ENOENT
    at errnoException (child_process.js:1000:11)
    at Process.ChildProcess._handle.onexit (child_process.js:791:34)

我该采取什么程序来修理呢?

作者注:这个错误的许多问题鼓励我发布这个问题,以供将来参考。

相关问题:

using spawn function with NODE_ENV=production node.js child_process.spawn ENOENT error - only under supervisord spawn ENOENT node.js error https://stackoverflow.com/questions/27603713/nodejs-spawn-enoent-error-on-travis-calling-global-npm-package Node JS - child_process spawn('npm install') in Grunt task results in ENOENT error Running "foreman" task Fatal error: spawn ENOENT unhandled error event in node js Error: spawn ENOENT at errnoException (child_process.js:975:11) Node.js SpookyJS: error executing hello.js https://stackoverflow.com/questions/26572214/run-grunt-on-a-directory-nodewebkit Run exe file with Child Process NodeJS Node: child_process.spawn not working on Java even though it's in the path (ENOENT) spawn ENOENT error with NodeJS (PYTHON related) image resizing is not working in node.js (partial.js) (non-installed dependency) npm install error ENOENT (build dependency problem) Cannot install node.js - oracle module on Windows 7 (build dependency problem) Error installing gulp using nodejs on windows (strange case)


当前回答

Windows解决方案:用节点交叉生成替换生成。例如,在你的app.js的开头:

(function() {
    var childProcess = require("child_process");
    childProcess.spawn = require('cross-spawn');
})(); 

其他回答

注意:此错误几乎总是由于命令不存在、工作目录不存在或由windows特有的错误引起的。

我发现了一个特别简单的方法来了解问题的根本原因:

Error: spawn ENOENT

这个错误的问题是,在错误消息中很少有信息告诉你调用的位置,即没有找到哪个可执行/命令,特别是当你有一个很大的代码库,其中有很多衍生调用时。另一方面,如果我们知道导致错误的确切命令,那么我们可以根据@laconbass的回答来解决问题。

我发现了一种非常简单的方法来发现导致问题的命令,而不是像@laconbass的回答中建议的那样在代码中到处添加事件侦听器。关键思想是用一个包装器包装原始的衍生调用,该包装器打印发送给衍生调用的参数。

这是包装器函数,把它放在index.js或任何你的服务器启动脚本的顶部。

(function() {
    var childProcess = require("child_process");
    var oldSpawn = childProcess.spawn;
    function mySpawn() {
        console.log('spawn called');
        console.log(arguments);
        var result = oldSpawn.apply(this, arguments);
        return result;
    }
    childProcess.spawn = mySpawn;
})();

然后,下次运行应用程序时,在未捕获异常的消息之前,您将看到如下内容:

spawn called
{ '0': 'hg',
  '1': [],
  '2':
   { cwd: '/* omitted */',
     env: { IP: '0.0.0.0' },
     args: [] } }

通过这种方式,你可以很容易地知道哪个命令实际上被执行了,然后你可以找出为什么nodejs不能找到可执行文件来解决这个问题。

你要改变环境选项吗?

然后看看这个答案。


我试图生成一个节点进程和TIL,您应该在生成时扩展现有的环境变量,否则您将失去PATH环境变量和其他可能重要的环境变量。

这就是我的解决方案:

const nodeProcess = spawn('node', ['--help'], {
  env: {
    // by default, spawn uses `process.env` for the value of `env`
    // you can _add_ to this behavior, by spreading `process.env`
    ...process.env,
    OTHER_ENV_VARIABLE: 'test',
  }
});

步骤1:确保以正确的方式调用spawn

首先,查看child_process的文档。衍生(命令,参数,选项):

使用给定的命令启动一个新进程,并使用args中的命令行参数。如果省略,args默认为空数组。 第三个参数用于指定额外的选项,默认为: {cwd:未定义,env:进程。env} 使用env指定新进程可见的环境变量,默认为process.env。

确保您没有在命令中放入任何命令行参数,并且整个衍生调用都是有效的。执行下一步。

步骤2:识别发出错误事件的事件发射器

在源代码中搜索每个spawn或child_process调用。产卵。

spawn('some-command', [ '--help' ]);

并为'error'事件附加一个事件监听器,这样你就会注意到确切的事件发射器将它作为'Unhandled'抛出。调试之后,可以删除该处理程序。

spawn('some-command', [ '--help' ])
  .on('error', function( err ){ throw err })
;

执行后,您将获得注册'error'侦听器的文件路径和行号。喜欢的东西:

/file/that/registers/the/error/listener.js:29
      throw err;
            ^
Error: spawn ENOENT
    at errnoException (child_process.js:1000:11)
    at Process.ChildProcess._handle.onexit (child_process.js:791:34)

如果前两行是静止的

events.js:72
        throw er; // Unhandled 'error' event

重复这一步,直到它们消失。在进行下一步之前,必须确定发出错误的侦听器。

步骤3:确保设置了环境变量$PATH

有两种可能的情况:

您依赖于默认的衍生行为,因此子进程环境将与process.env相同。 你是显式地传递一个env对象到options参数上产卵。

在这两种场景中,都必须检查生成的子进程将使用的环境对象上的PATH键。

场景1的示例

// inspect the PATH key on process.env
console.log( process.env.PATH );
spawn('some-command', ['--help']);

场景2的示例

var env = getEnvKeyValuePairsSomeHow();
// inspect the PATH key on the env object
console.log( env.PATH );
spawn('some-command', ['--help'], { env: env });

PATH的缺失(即,它是未定义的)将导致spawn发出ENOENT错误,因为它不可能定位任何命令,除非它是可执行文件的绝对路径。

当PATH设置正确后,执行下一步。它应该是一个目录或目录列表。最后一种情况很常见。

步骤4:确保命令存在于PATH中定义的目录中

如果filename命令(即'some-command')在PATH上定义的至少一个目录中不存在,Spawn可能会发出ENOENT错误。

找到命令的确切位置。在大多数linux发行版上,这可以在终端上使用which命令完成。它会告诉你可执行文件的绝对路径(如上所述),或者告诉你是否没有找到它。

找到命令时which的用法示例及其输出

> which some-command
some-command is /usr/bin/some-command

在找不到命令时,which及其输出的示例用法

> which some-command
bash: type: some-command: not found

未安装的程序是导致找不到命令的最常见原因。如果需要,请参考每个命令文档并安装它。

当命令是一个简单的脚本文件时,确保它可以从PATH上的目录访问。如果不是,要么把它移到一个,要么建立一个链接。

一旦你确定PATH被正确设置并且命令可以从它访问,你应该能够在不抛出spawn ENOENT的情况下生成你的子进程。

我在windows 8上得到了同样的错误。问题是由于系统路径的环境变量丢失。将“C:\Windows\System32\”值添加到系统PATH变量中。

我在Windows上遇到了这个问题,用完全相同的命令(省略参数)调用exec和spawn对于exec来说工作得很好(所以我知道我的命令在$PATH上),但spawn会给出ENOENT。原来我只需要将.exe附加到我正在使用的命令:

import { exec, spawn } from 'child_process';

// This works fine
exec('p4 changes -s submitted');

// This gives the ENOENT error
spawn('p4');

// But this resolves it
spawn('p4.exe');
// Even works with the arguments now
spawn('p4.exe', ['changes', '-s', 'submitted']);