验证是否安装了帮助工具

Verify that a helper tool is installed

我正在 Swift 中编写一个需要特权帮助工具的 macOS 应用程序 -- 希望提升不是必需的,但是 it looks like it is.

我发现 this 非常适合这种情况的示例应用程序。我已经设法将它的代码移植到我自己的应用程序中,但我被困在需要检查是否安装了辅助工具的地步,如果没有,请使用 SMJobBless() 和朋友安装它。

当运行示例应用程序时,如果没有安装辅助工具,应用程序将停留在以下屏幕:

明确地说,通过阅读代码,我认为它应该在某个时候将标签更新为 "Helper Installed: No",但这似乎没有发生。

如果我点击 "Install Helper",这就是结果。

从现在开始,除非我手动删除辅助工具,否则应用程序将运行显示此屏幕 "Helper Installed: Yes"。

在用户必须手动单击 "Install Helper" 按钮的示例情况下,此行为可能没问题。但是,在我的应用程序中,如果尚未安装,我希望它自动请求安装帮助工具。如果已经安装,我不想浪费用户的时间再次请求他们的密码。

我认为这很简单:如果辅助工具不可用,在连接过程中的某个地方会发生错误,这就是我请求安装该工具的触发因素。如果没有错误发生,则认为该工具已经安装。

这是我编写的用于通过 XPC 连接到辅助工具的破解代码:

var helperConnection: NSXPCConnection?
var xpcErrorHandler: ((Error) -> Void)?
var helper: MyServiceProtocol?

// ...

helperConnection = NSXPCConnection(machServiceName: MyServiceName, options: .privileged)
helperConnection?.remoteObjectInterface = NSXPCInterface(with: MyServiceProtocol.self)
helperConnection?.resume()

helperConnection?.interruptionHandler = {
    // Handle interruption
    NSLog("interruptionHandler()")
}

helperConnection?.invalidationHandler = {
    // Handle invalidation
    NSLog("invalidationHandler()")
}

xpcErrorHandler = { error in
   NSLog("xpcErrorHandler: \(error.localizedDescription)")
}

guard
    let errorHandler = xpcErrorHandler,
    let helperService = helperConnection?.remoteObjectProxyWithErrorHandler(errorHandler) as? MyServiceProtocol
    else {
        return
}

helper = helperService

如果未安装帮助工具,运行 此代码不会产生错误或 NSLog() 输出。如果之后我通过 XPC 调用一个函数(使用 helper?.someFunction(...)),什么也不会发生——我还不如与 /dev/null.

交谈

现在我正在抓耳挠腮地寻找一种技术来检测该工具是否已安装。示例应用程序解决问题的方法是添加一个 getVersion() 方法;如果它 returns 某些东西,"Install Helper" 变灰并且标签变为 "Helper Installed: Yes"。

我考虑通过在我的工具中编写一个简单的函数来稍微扩展这个想法 returns 立即,并在主应用程序中使用超时——如果我在代码之前没有得到结果超时,帮助工具可能未安装。我发现这是一个棘手的解决方案——例如,如果辅助工具(按需启动)启动时间有点太长,比如因为计算机很旧并且用户是 运行 东西 CPU密集?

我看到了其他替代方法,例如在预期位置(/Library/PrivilegedHelperTools/Library/LaunchDaemons)查看文件系统,但我还是觉得这个解决方案不令人满意。

我的问题:有没有办法明确检测特权 XPC 帮助工具是否在另一端侦听?

我的环境:macOS Mojave 10.14.2,Xcode 10.1,Swift 4.2.

我会检查文件系统是否存在二进制文件(在 /Library/PrivilegedHelperTools 中,如果 plist 存在于 /Library/LaunchDaemons 中)。然后你可能会联系 XPC 服务并调用某种 ping 功能,如果服务已启动并且 运行.

只是我的 2 克拉,

罗伯特

由于您创建了辅助工具,只需添加一个 XPC 消息处理程序来报告您的工具的状态。当您启动时,连接并发送该消息。如果其中任何一个失败,则说明您的工具未正确安装(或没有响应)。

在我的代码中,我的所有 XPC 服务(包括我的特权助手)都采用了用于测试和操作安装的基本协议:

@protocol DDComponentInstalling /*<NSObject>*/

@required
- (void)queryBuildNumberWithReply:(void(^_Nonnull)(UInt32))reply;

@optional
- (void)didInstallComponent;
- (void)willUninstallComponent;

queryBuildNumberWithReply: returns 一个描述组件版本号的整数:

- (void)queryBuildNumberWithReply:(void(^)(UInt32))reply
{
    reply(FULL_BUILD_VERSION);
}

如果消息成功,我将返回值与应用程序中的内部版本号常量进行比较。如果不匹配,则该服务是 older/newer 版本,需要更换。对于我的产品的每个 public 版本,此常量都会增加。

我使用的代码如下所示:

- (BOOL)verifyServiceVersion
{
    DDConnection* connection = self.serviceConnection;
    id<DDComponentInstalling> proxy = connection.serviceProxy;  // get the proxy (will connect, as needed)
    if (proxy==nil)
        // an XPC connection could not be established or the proxy object could not be obtained
        return NO;  // assume service is not installed

    // Ask for the version number and wait for a response
    NSConditionLock* barrierLock = [[NSConditionLock alloc] initWithCondition:NO];
    __block UInt32 serviceVersion = UNKNOWN_BUILD_VERSION;
    [proxy queryBuildNumberWithReply:^(UInt32 version) {
        // Executes when service returns the build version
        [barrierLock lock];
        serviceVersion = version;
        [barrierLock unlockWithCondition:YES];  // signal to foreground thead that query is finished
        }];
    // wait for the message to reply
    [barrierLock lockWhenCondition:YES beforeDate:[NSDate dateWithTimeIntervalSinceNow:30.0];
    BOOL answer = (serviceVersion==FULL_BUILD_VERSION); // YES means helper is installed, alive, and correct version
    [barrierLock unlock];

    return answer;
}

请注意,DDConnection 是 XPC 连接的实用程序包装器,barrierLock 技巧实际上封装在一个共享方法中——所以我不会一遍又一遍地写这个——但是出于演示目的,此处展开。

我还有 pre/post-install/upgrade 问题要处理,所以我的所有组件都实现了一个可选的 didInstallComponentwillUninstallComponent 方法,我在安装新助手后立即发送,或者只是在我打算卸载或替换安装的助手之前。

背景

实际上可以避免等待工具响应超时。事实上,如果未启用该工具(参见 here),您引用的 erikberglund/SwiftPrivilegedHelper 示例确实会在“Helper Installed”文本字段旁边打印“No”一词,诚然是异步的,但实际上是即时的。

但是,我处于一个独特的位置,我遇到了你在我自己的特权助手实现中描述的问题,但是 SwiftPrivilegedHelper 示例完全适合我。未安装该工具时,前者永远不会调用 remoteObjectProxyWithErrorHandler 错误处理程序,而后者会调用。

因为我有一个关于您的问题的示例以及一个工作示例,我相信我已经设法找出根本原因:

根本原因

至少就我而言,我 没有彻底卸载辅助工具

在某些时候,我从 /Library/LaunchDaemons 中删除了它的 plist,从 /Library/PrivilegedHelperTools 中删除了工具本身,但我想我没有 运行 sudo launchctl unload /Library/LaunchDaemons/com.example.foo.plist

我运行sudo launchctl list,或sudo launchctl print system/com.example.foo.plist.

的时候还列出来了

在这种情况下,调用该工具显然不会成功(因为该工具未安装)但不会失败(因为 launchctl 认为它 安装) .

解决方案

可以通过以下两种方式之一进行正确的卸载:

  1. sudo launchctl remove com.example.foo(注:不是system/com.example.foo
  2. sudo launchctl unload /Library/LaunchDaemons/com.example.foo.plist(这要求 plist 仍然存在于磁盘上)。

如果正确卸载,sudo launchctl print system/ca.example.foo 应该打印:

Bad request.
Could not find service "com.example.foo" in domain for system

Quinn “The Eskimo!” suggests the following sequence,它使用 remove 方法:

  1. Remove property list.
  2. Remove helper tool.
  3. Remove the job.

我一清理完东西,你瞧,我的 remoteObjectProxyWithErrorHandler 在未安装该工具时开始调用其错误处理程序。