0

使用具有以下 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 为什么会这样。

4

1 回答 1

1

启动守护程序(以及 cron 作业和...)不会在您通常的终端/shell 环境下运行,因此它们不会在您的终端/shell 环境中设置任何自定义、附加组件等。特别是,它们不运行通常具有设置环境的命令的各种 shell 初始化脚本(~/.bash_profile、~/.bashrc 等)。

在 anaconda 的特定情况下,它的安装程序会在您的 ~/.bashrc 文件中添加一个部分,以将 anaconda 二进制文件目录添加到PATH,并可能进行一些其他更改。这会影响您在终端中的 bash 会话环境,但不会对启动代理生效,甚至对其他 shell 中的终端会话都不起作用(zsh 现在是默认设置,这会造成一些问题)。

对于启动守护程序,我建议不要让它运行您的 shell 设置脚本——这些是您的个人配置,并且没有任何实际业务由系统进程运行。相反,我会创建一个简短的 shell 脚本,其中包含相关设置(从您的 ~/.bashrc 复制),然后运行您的 python 脚本。然后更改启动守护程序,使其运行该脚本,而不是直接运行 python 脚本。

PS由于脚本在启动python脚本后不需要做任何事情,这是脚本可以使用exec命令退出自身,并在同一进程中运行python脚本的情况。这意味着python将是launchd的直接子进程,并且launchd可以以它喜欢的方式对其进行监视、控制等。(这与让脚本正常运行 python 不同,在这种情况下,您将有一个现在毫无意义的 shell 进程挂起等待 python 脚本完成,以便它可以完成。)

所以我会用以下方式结束脚本:

exec python3 /Users/my-name/python-daemon/python_daemon_test.py
于 2020-04-25T21:07:29.417 回答