【问题标题】:pm2 managed process in errored state after machine reboot机器重启后 pm2 托管进程处于错误状态
【发布时间】:2023-12-10 17:59:01
【问题描述】:

我使用pm2(3.4.1 版)来管理Laravel Echo websocket server。它运行完美,但我试图让它在机器重启后自动启动(运行 Ubuntu 16.04)。为此,我关注了these instructions

$ pm2 startup
[PM2] Init System found: systemd
[PM2] To setup the Startup Script, copy/paste the following command:
sudo env PATH=$PATH:/usr/bin /usr/local/share/.config/yarn/global/node_modules/pm2/bin/pm2 startup systemd -u kramer65 --hp /home/kramer65

$ sudo env PATH=$PATH:/usr/bin /usr/local/share/.config/yarn/global/node_modules/pm2/bin/pm2 startup systemd -u kramer65 --hp /home/kramer65
[sudo] password for kramer65: 
[PM2] Init System found: systemd
Platform systemd
Template
[Unit]
Description=PM2 process manager
Documentation=https://pm2.keymetrics.io/
After=network.target

[Service]
Type=forking
User=kramer65
LimitNOFILE=infinity
LimitNPROC=infinity
LimitCORE=infinity
Environment=PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/usr/local/games:/usr/bin:/bin:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin
Environment=PM2_HOME=/home/kramer65/.pm2
PIDFile=/home/kramer65/.pm2/pm2.pid

ExecStart=/usr/local/share/.config/yarn/global/node_modules/pm2/bin/pm2 resurrect
ExecReload=/usr/local/share/.config/yarn/global/node_modules/pm2/bin/pm2 reload all
ExecStop=/usr/local/share/.config/yarn/global/node_modules/pm2/bin/pm2 kill

[Install]
WantedBy=multi-user.target

Target path
/etc/systemd/system/pm2-kramer65.service
Command list
[ 'systemctl enable pm2-kramer65' ]
[PM2] Writing init configuration in /etc/systemd/system/pm2-kramer65.service
[PM2] Making script booting at startup...
[PM2] [-] Executing: systemctl enable pm2-kramer65...
[PM2] [v] Command successfully executed.
+---------------------------------------+
[PM2] Freeze a process list on reboot via: $ pm2 save

[PM2] Remove init script via: $ pm2 unstartup systemd

然后我跑了pm2 save

$ pm2 save
[PM2] Saving current process list...
[PM2] Successfully saved in /home/kramer65/.pm2/dump.pm2

在此之后我重新启动机器,重新 ssh 并检查 pm2 进程:

$ pm2 status
┌──────────┬────┬─────────┬──────┬─────┬─────────┬─────────┬────────┬─────┬────────┬────────┬──────────┐
│ App name │ id │ version │ mode │ pid │ status  │ restart │ uptime │ cpu │ mem    │ user   │ watching │
├──────────┼────┼─────────┼──────┼─────┼─────────┼─────────┼────────┼─────┼────────┼────────┼──────────┤
│ echo     │ 0  │ N/A     │ fork │ N/A │ errored │ 0       │ 0      │ 0%  │ 0 B    │ kramer65 │ disabled │
└──────────┴────┴─────────┴──────┴─────┴─────────┴─────────┴────────┴─────┴────────┴────────┴──────────┘
 Use `pm2 show <id|name>` to get more details about an app

如您所见,状态为错误。如果我停止并启动进程 (pm2 stop echo &amp;&amp; pm2 start echo),进程会再次运行。

错误日志 (~/.pm2/pm2.log) 告诉我:

2019-05-14T11:56:36: PM2 log: ===============================================================================
2019-05-14T11:56:36: PM2 log: --- New PM2 Daemon started ----------------------------------------------------
2019-05-14T11:56:36: PM2 log: Time                 : Tue May 14 2019 11:56:36 GMT+0200 (CEST)
2019-05-14T11:56:36: PM2 log: PM2 version          : 3.4.1
2019-05-14T11:56:36: PM2 log: Node.js version      : 8.15.1
2019-05-14T11:56:36: PM2 log: Current arch         : x64
2019-05-14T11:56:36: PM2 log: PM2 home             : /home/kramer65/.pm2
2019-05-14T11:56:36: PM2 log: PM2 PID file         : /home/kramer65/.pm2/pm2.pid
2019-05-14T11:56:36: PM2 log: RPC socket file      : /home/kramer65/.pm2/rpc.sock
2019-05-14T11:56:36: PM2 log: BUS socket file      : /home/kramer65/.pm2/pub.sock
2019-05-14T11:56:36: PM2 log: Application log path : /home/kramer65/.pm2/logs
2019-05-14T11:56:36: PM2 log: Process dump file    : /home/kramer65/.pm2/dump.pm2
2019-05-14T11:56:36: PM2 log: Concurrent actions   : 2
2019-05-14T11:56:36: PM2 log: SIGTERM timeout      : 1600
2019-05-14T11:56:36: PM2 log: ===============================================================================
2019-05-14T11:56:36: PM2 log: App [echo:0] starting in -fork mode-
2019-05-14T11:56:36: PM2 log: App [echo:0] online
2019-05-14T11:56:36: PM2 error: Error: spawn node ENOENT
    at Process.ChildProcess._handle.onexit (internal/child_process.js:190:19)
    at onErrorNT (internal/child_process.js:362:16)
    at _combinedTickCallback (internal/process/next_tick.js:139:11)
    at process._tickDomainCallback (internal/process/next_tick.js:219:9)

有人知道这里出了什么问题吗?

【问题讨论】:

  • journalctl -u service-name.service 说什么?
  • 你的 pm2 版本是多少?
  • @suv - 你的意思是字面意义上的journalctl -u service-name.service?这就是No journal files were opened due to insufficient permissions.,如果我用sudo 调用它,我会得到-- No entries --。我假设您的意思是我必须将 service-name 更改为其他内容。
  • @MehdiBenmoha - 我正在使用 pm2 版本 3.4.1。我也添加了这个问题。
  • @kramer65 - 不。不是字面意思。使用服务名称调用命令。很可能是 pm2-kramer65.service 所以它变成了journalctl -u pm2-kramer65.service

标签: javascript pm2 laravel-echo process-management


【解决方案1】:

ENOENT 代表No such directory entry

env &lt;command&gt; 在做什么? (by Chris Johnsen)

env command 用于在自定义环境中运行另一个程序,而无需修改当前程序。

sudo env PATH=$PATH:/usr/bin \
   /usr/local/share/.config/yarn/global/node_modules/pm2/bin/pm2 \
   startup systemd -u kramer65 --hp /home/kramer65

安装来自pm2 documentation的注释 要检测机器上可用的 init 系统并生成配置,请使用:

pm2 startup
$ [PM2] You have to run this command as root. Execute the following command:
$ sudo su -c env PATH=$PATH:/home/unitech/.nvm/versions/node/v4.3/bin pm2 startup <distribution> -u <user> --hp <home-path>

我将命令分解如下

sudo env name=value name2=value2 program and args

1) 计划

运行命令程序 /usr/local/share/.config/yarn/global/node_modules/pm2/bin/pm2

剩下的第一个参数指定要调用的程序名称;它是根据 `PATH' 环境变量搜索的。

2) 参数通过

任何剩余的参数都作为参数传递给该程序。

传递给pm2 cli 命令的argsstartup systemd -u kramer65 --hp /home/kramer65

  • startups 命令将简单地将 pm2 设置为在引导时启动 (Startup Cli Commands)
  • systemd 命令用于受支持的初始化系统 Ubuntu =&gt; 16, CenOS &gt;=7, Arch, Debian &gt;=7 (Startup Hooks Compatability),应按照包含的文档更改为正确的 platform

3) 用于执行程序的环境

具有通过使用 name=value 和 name2=value2 指定的环境变量和值扩展当前环境而形成的环境。

pm2 文档应该是/home/kramer/.nvm/versions/node/v4.3/bin 内的文件夹

PATH=$PATH:/home/unitech/.nvm/versions/node/v4.3/bin

当您使用 home 目录之外的文件夹时,因为您在 nodejs 上运行旧版本(参见上一章 Nodejs 安装在 usr/bin/node)。 p>

PATH=$PATH:/usr/bin

4) 用户权限

您正在将--user kramer65 --home-path /home/kramer65 传递给pm2 cli 命令。

-u <user> --hp <home-path>

保存您的流程

启动挂钩会自动加载您之前保存的进程列表。 保存您的进程列表:

pm2 save

Nodejs 安装在usr/bin/node (Error: spawn ENOENT on Ubuntu 14.04)

如何重现:您的计算机上有两个版本的 nodejs。一个是安装在/usr/bin/node 的旧版本,另一个是位于./nodejs/node 的最新版本。我通过调用 ./nodejs/node myManager.js 以编程方式启动我的 pm2 管理器,这将调用 pm2.connect(),它将尝试生成 PM2 守护进程。

发生的情况是 PM2 守护程序将使用来自 /usr/bin/node 的旧节点二进制文件而不是运行我的 myManager.js 脚本(位于 ./nodejs/node)的最新节点二进制文件启动。

这样做是在当前目录中查找节点二进制文件,然后默认查找路径中的节点二进制文件。 但是运行当前代码的节点二进制文件位于./nodejs/node。要使用与当前进程相同的节点二进制文件启动 PM2 守护进程,您需要使用 process.execPath

你可以阅读more info at in the relevant discussion

【讨论】:

  • 感谢您的广泛回答。我花了一些时间来理解它。我只是不明白你对the other one is a recent version located in ./nodejs/node 的意思。那是相对路径。我在我的系统 (sudo find / -name *node*) 上进行了搜索,但没有找到任何其他节点版本。或者它只是在寻找它没有找到的另一个节点版本。你能给我更多关于这可能出错的提示吗?
  • @kramer65 抱歉,我坐了 18 小时的航班。我正在通过移动设备接听电话。我引用了一个讨论中的消息。似乎在 y 子进程中找不到 /usr/local/share/.config/yarn/global/node_modules/pm2/bin/pm2 的文件夹? Y 正在传递 user home --hp /home/kramer 但 yarn 正在安装 y 主文件夹之外的依赖项。我会考虑升级纱线和节点
  • 我不认为问题出在--hp 选项上,但您的systemmd 没有从pm2 路径中找到文件夹
  • 在更好地了解 systemmd 的工作原理、在 systemmd 中生成终端会话并测试这些命令后,可以解决此问题