LaunchDaemon在终端上运行不同的Python实例



使用具有以下plist的LaunchDaemon:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>Label</key>
<string>com.test.testing</string>
<key>ProgramArguments</key>
<array>
<string>python3</string>
<string>/Users/my-name/python-daemon/python_daemon_test.py</string>
</array>
<key>StandardErrorPath</key>
<string>/var/log/test-Error.log</string>
<key>StandardOutPath</key>
<string>/var/log/test.log</string>
<key>RunAtLoad</key>
<true/>
</dict>
</plist>

要启动的脚本是

#!/usr/bin/env python3
import sys
print(sys.version)
print(sys.path)

我的行为我不理解。

当我在终端中运行脚本时(其中which python3导致/Users/my-name/opt/anaconda3/bin/python3(,我收到以下输出:

3.7.4 (default, Aug 13 2019, 15:17:50)
[Clang 4.0.1 (tags/RELEASE_401/final)]
['/Users/my-name/python-daemon', '/Users/my-name/opt/anaconda3/lib/python37.zip', 
'/Users/my-name/opt/anaconda3/lib/python3.7', 
'/Users/my-name/opt/anaconda3/lib/python3.7/lib-dynload', 
'/Users/my-name/.local/lib/python3.7/site-packages', 
'/Users/my-name/opt/anaconda3/lib/python3.7/site-packages', 
'/Users/my-name/opt/anaconda3/lib/python3.7/site-packages/aeosa']

这是所需的,因为我希望运行的实际(非玩具(脚本使用了我在Anaconda中安装的PyObjc包。

但是,当LaunchDaemon运行脚本时,我在test.log文件中得到以下内容:

3.7.3 (default, Mar  6 2020, 22:34:30) 
[Clang 11.0.3 (clang-1103.0.32.29)]
['/Users/my-name/python-daemon', 
'/Applications/Xcode.app/Contents/Developer/Library/Frameworks/Python3.framework/Versions/3.7/lib/python37.zip', 
'/Applications/Xcode.app/Contents/Developer/Library/Frameworks/Python3.framework/Versions/3.7/lib/python3.7', 
'/Applications/Xcode.app/Contents/Developer/Library/Frameworks/Python3.framework/Versions/3.7/lib/python3.7/lib-dynload', 
'/Applications/Xcode.app/Contents/Developer/Library/Frameworks/Python3.framework/Versions/3.7/lib/python3.7/site-packages']

这是一个问题,因为我希望运行的实际脚本取决于Anaconda中的一个包。我本来以为,由于脚本中的第一个程序参数是python3,它将使用与终端中相同的python3运行,但我错了。

为什么剧本会这样?我认为这可能是因为它作为sudo运行脚本,但sudo which python3也返回Anaconda的路径。

我该如何解决这个问题?我相信一个简单的解决方案就是在Python3的Xcode版本中安装我想要的软件包。然而,我真的不知道该怎么做。这也会让人觉得有点不满意,因为我想知道为什么LaunchDaemons会这样做。

Launch Daemon(以及cron jobs和…(不在您通常的Terminal/shell环境下运行,因此它们不会获得在您的Terminal/sshell环境中设置的任何自定义、加载项等。特别是,它们不运行各种shell初始化脚本(~/.bash_profile、~/.bahrc等(,这些脚本通常具有设置环境的命令。

在anaconda的特定情况下,它的安装程序会在~/.bashrc文件中添加一个部分,将anaconda二进制文件目录添加到PATH,可能还会添加一些其他更改。这会影响你的环境——终端中的bash会话,但不会对Launch Agent生效,甚至不会对其他shell中的终端会话生效(zsh现在是默认值,这会带来一些麻烦(。

对于Launch Daemon,我建议不要让它运行你的shell设置脚本——这些是你的个人配置,没有任何真正的业务由系统进程运行。相反,我会创建一个包含相关设置的短shell脚本(从~/.bashrc复制(,然后运行python脚本。然后更改Launch Daemon,使其运行该脚本,而不是直接运行python脚本。

p.S.由于脚本在启动python脚本后不必执行任何操作,因此在这种情况下,脚本可以使用exec命令退出自身,并在同一进程中运行python脚本。这意味着python将是launchd的直接子进程,launchd可以按照自己喜欢的方式对其进行监视、控制等

所以我会以结束脚本

exec python3 /Users/my-name/python-daemon/python_daemon_test.py

最新更新