检测 macOS 应用程序是否从命令行(终端)启动

Detect whether macOS app was launched from the command line (Terminal)

我有一个 GUI macOS 应用程序,它也可以从终端启动,带有可选的命令行参数。

当使用参数启动时,我喜欢 运行 应用程序处于 "cmdline" 模式,在这种模式下我不显示任何 UI 而是仅通过 stdin + stdout 进行通信。

我可以像这样检测此 cmdline 模式:

BOOL cmdMode = NSProcessInfo.processInfo.arguments.count > 1;

(arg 0 始终是可执行文件的路径,因此将手动传递更多的 args)。

现在,问题来了:

如果用户调用我的应用程序 没有来自终端的 参数(通过在 Contents/MacOS 中调用应用程序的可执行文件,即不通过 open cmd),我也喜欢进入 cmdline 模式。我该如何检测?

注意:较旧的 OS X 版本确实传递了一个“-psn ...”参数,当不存在时,可用于检测从 cmdline 启动,但最近的 macOS从 Finder 启动应用程序时,版本似乎不再传递此参数,因此我不能再使用它进行检测。

更新

我意识到我几乎可以通过检查某些环境变量的存在来正确解决这个问题:

TERMPWD 仅在从终端而不是从 Finder 启动应用程序时设置。

但是,我也希望能够区分直接启动(可在 Contents/MacOS 目录中执行)与使用 open 命令启动之间的区别,因为我认为 open cmd 是等效的通过 Finder 打开应用程序或通过 Launch Services 从另一个应用程序打开应用程序。

简而言之,问题也可能是:检测应用程序是否由 Launch Services 启动


作为记录,这里是 environ() 中的值。标有星号的那些仅在从 Terminal.app 调用时存在,但在从 Finder 启动时不存在:

    __CF_USER_TEXT_ENCODING=0x1F5:0x0:0x0
*   _=/Applications/Myapp.app/Contents/MacOS/Myapp
    Apple_PubSub_Socket_Render=/private/tmp/com.apple.launchd.laVQnD7IXl/Render
    HOME=/Users/username
*   LANG=en_US.UTF-8
*   LC_ALL=en_US.UTF-8
*   LC_CTYPE=UTF-8
    LOGNAME=username
    PATH=/usr/local/sbin:/usr/local/bin:/usr/bin:/bin:/usr/sbin:/sbin
*   PWD=/Users/username
    SHELL=/bin/bash
*   SHLVL=1
    SSH_AUTH_SOCK=/private/tmp/com.apple.launchd.KeHv8KNuuk/Listeners
*   TERM_PROGRAM_VERSION=388.1.2
*   TERM_PROGRAM=Apple_Terminal
*   TERM_SESSION_ID=EF2C59E6-D661-45BE-B7EF-7A0E71158C8D
*   TERM=xterm-color
    TMPDIR=/var/folders/hm/ycnxcbwx8xl1v7008k8wnpjh0000gn/T/
    USER=username
    XPC_FLAGS=0x0
    XPC_SERVICE_NAME=0

但是,对于使用 Launch Services 启动的应用程序(例如在 Finder 中双击时),没有唯一的环境值。

如果你想知道哪个进程执行了你的程序,你可以使用getppid()获取父进程ID,然后检查该进程以确定你是否被交互式执行shell process,或Finder,或launchctl等

/sbin/launchd 是 PID 1 - 如果你的进程的父 PID 是 1,你被 launchd 执行了。

否则,您将被另一个进程执行 - 可能是交互式 shell,或者作为另一个进程的子进程。您可以使用 KERN_PROCARGS syscall with sysctl() 通过其 PID 获取进程名称。

您可能还想考虑使用 isatty(STDIN):交互式 shells 有 TTY,非交互式 shells 而其他进程则没有。