大家都知道,php的proc_open功能很强大,能和外部程序交互。
而我做的工作是一个结算中心。涉及各种虚拟货币的结算和发款。什么 比特币btc 莱特币ltc 门罗xmr 等各种币
他们之间有相同,也有各自不同的地方,发款之前都是用rpc交互,本没有什么问题。
但是门罗系列的xmr xmc etn 等,rpc不好用,且币种问题,rpc太慢了,所以我采用了 cli 的交互。
后来又有 以太坊eth 的发款,它又是一个已经写好的nodejs脚本,刚好也能用 proc_open 去做。
接下来就是问题了,我在dockerfile内,写了 nvm、nodejs、yarn 的安装,启动后的容器也正常。
但是测试脚本通不过。发现在容器内使用 proc_open 调用 nodejs 脚本的时候,从流里面读不到数据,但是明明我在mac上是测试通过的。并且最头大的情况是,没有任何异常,只是进程一直hand住。
我中途想到了权限问题,测试不过。想到了cwd的参数,测试不过。
最后发现了问题所在,我docker容器内,整个Web服务,nginx,phpfpm等,都是运行在www-data用户下。当我的web接收到发款指令后 ,通过proc_open 出来的进程也是www-data的,而我的node是最开始dockerfile内通过nvm安装,所以是装在root用户下的。 气人的是,你可以在容器内尝试 sudo -u www-data node -v ,会提示你找不到node命令。但是proc_open没有任何错误提示。
简单来说,我的情况就是 运行着的php脚步通过proc_open调用了一个不存在的命令,这个进程不会挂,不会有输出。
最后我发现 proc_open去调用任何不存在的命令都不会提示“zsh: command not found: XXXX”, 而是会得到feof,所以是我没有正确去使用。
目前我是保证docker内的node是所有用户可用的,就解决了,但是feof的问题还是应该去处理。