Lazarus Pascal:使用SMJobBless()编写特权帮助工具

问题描述 投票:1回答:2

我在Lazarus Pascal中创建了一个应用程序,它执行“dd”将图像写入驱动器。为此,显然需要提升权利。

在最初的版本中,我使用了AuthorizationExecuteWithPrivileges()link),尽管它并非完全用于此目的,但它确实非常有效并且非常一致。但是,自OSX 10.7起,此函数已被折旧,因为它可能是一个安全问题,并且重定向的命令行语句也无法正常工作(将zip的输出重定向为dd的输入)。

在下一个版本中,我使用了Lazarus Pascal Wiki(Executing External Programs)中描述的方法,该方法基本上启动了我的程序与之通信的TProcess。使用sudo -S dd ...,会询问并输入用户密码以确保他/她具有适当的访问权限。显然,这是一种肮脏的黑客方法,它表明,某些用户遇到了这方面的问题。

在做了很多阅读之后,似乎Apple更喜欢使用名为SMJobBless()的辅助工具。遗憾的是,就Objective-C而言,我并不是很有经验,所提出的代码看起来非常简单,也没有很好的记录。

我想知道是否有人有经验或可以帮助将这种方法“移植”到Lazarus Pascal ......我全都赞成做得对。当然,也欢迎其他方法!

任何帮助将不胜感激。

macos root lazarus access-rights smjobbless
2个回答
1
投票

遗憾的是,就Objective-C而言,我并不是很有经验

不要让这让你不再使用Apple提供的示例。如果你仔细查看SMJobBlessAppController.m中的代码,你会发现除了一行Objective-C代码之外,其余的只是C.

Objective-C行注册帮助应用程序: -

if (![self blessHelperWithLabel:@"com.apple.bsd.SMJobBlessHelper" error:&error])

您将使用自己的URI,而不是com.apple.bsd.SMJobBlessHelper。

所有其他相关的行都是普通的C函数。打破这一点,你只剩下: -

// Obtain rights 
AuthorizationCopyRights(self->_authRef, &authRights, kAuthorizationEmptyEnvironment, flags, NULL)


//Start the helper
SMJobBless(kSMDomainSystemLaunchd, (CFStringRef)label, self->_authRef, &cfError);

我没有检查错误代码,但我希望这表明你需要使用的代码很少,而且需要很少的Objective-C知识。


0
投票

因为它花了我很多工作,并认为这对其他人有帮助,这是我最后的工作解决方案。 https://www.tweaking4all.com/software-development/lazarus-development/macos-smjobbless-elevated-privileges-lazarus-pascal/

你会发现一个示例项目和大量的信息。

重现这一点的步骤相当广泛,所以这里简单回顾一下:

我一直在使用CFMessages向Helper工具发送消息,因为我没有绑定NSXPCConnection。

Helper Tool必须基于Lazarus Pascal模板“程序”或“简单程序”,并且不能基于任何TApplication类,也不能创建任何踏板。对于Helper工具,需要创建一个info.plist和一个launchd.plist,它们都必须嵌入到二进制文件中。

Main(测试)应用程序可以是任何Lazarus Pascal应用程序,但也需要适当的Info.plist,表明允许Helper Tool以提升的权限启动。

Helper Tool和应用程序应用程序包都需要使用有效的Apple Developer ID进行签名。

一些缺少的绑定需要到位:

const  kSMRightBlessPrivilegedHelper = 'com.apple.ServiceManagement.blesshelper';
function SMJobBless(domain:CFStringRef; executableLabel:CFStringRef; auth:AuthorizationRef; outError:CFErrorRef): boolean; external name '_SMJobBless'; mwpascal; 
var kSMDomainSystemLaunchd: CFStringRef; external name '_kSMDomainSystemLaunchd';

并且需要包含适当的框架:

{$linkframework ServiceManagement}
{$linkframework Security}
{$linkframework Foundation}
{$linkframework CoreFoundation}
{$calling mwpascal}

让我们不要忘记设置回调函数来处理传入的消息。

我希望这对某人有用...... :-)

© www.soinside.com 2019 - 2024. All rights reserved.