9

我的目标是创建一个执行 clang-format 的扩展。我的代码看起来像这样:

- (void)performCommandWithInvocation:(XCSourceEditorCommandInvocation *)invocation completionHandler:(void (^)(NSError * _Nullable nilOrError))completionHandler
{
    NSError *error = nil;

    NSURL *executableURL = [[self class] executableURL];

    if (!executableURL)
    {
          NSString *errorDescription = [NSString stringWithFormat:@"Failed to find clang-format. Ensure it is installed at any of these locations\n%@", [[self class] clangFormatUrls]];
              completionHandler([NSError errorWithDomain:SourceEditorCommandErrorDomain
              code:1
              userInfo:@{NSLocalizedDescriptionKey: errorDescription}]);
          return;
    }

    NSMutableArray *args = [NSMutableArray array];
    [args addObject:@"-style=LLVM"];
    [args addObject:@"someFile.m"];
    NSPipe *outputPipe = [NSPipe pipe];
    NSPipe *errorPipe = [NSPipe pipe];

    NSTask *task = [[NSTask alloc] init];
    task.launchPath = executableURL.path;
    task.arguments = args;

    task.standardOutput = outputPipe;
    task.standardError = errorPipe;

    @try
    {
          [task launch];
    }
    @catch (NSException *exception)
    {
          completionHandler([NSError errorWithDomain:SourceEditorCommandErrorDomain
              code:2
              userInfo:@{NSLocalizedDescriptionKey: [NSString stringWithFormat:@"Failed to run clang-format: %@", exception.reason]}]);
          return;
    }

    [task waitUntilExit];

    NSString *output = [[NSString alloc] initWithData:[[outputPipe fileHandleForReading] readDataToEndOfFile]
          encoding:NSUTF8StringEncoding];
    NSString *errorOutput = [[NSString alloc] initWithData:[[errorPipe fileHandleForReading] readDataToEndOfFile]
          encoding:NSUTF8StringEncoding];
    [[outputPipe fileHandleForReading] closeFile];
    [[errorPipe fileHandleForReading] closeFile];

    int status = [task terminationStatus];
    if (status == 0)
    {
          NSLog(@"Success: %@", output);
    }
    else
    {
          error = [NSError errorWithDomain:SourceEditorCommandErrorDomain
              code:3
              userInfo:@{NSLocalizedDescriptionKey: errorOutput}];
    }

    completionHandler(error);
}

我需要 try-catch 块的原因是当我尝试运行此代码时引发异常。异常原因是:

错误:无法访问启动路径

我的 clang-format 的路径是 /usr/local/bin/clang-format。我发现它不喜欢我尝试访问/usr/local/bin 中的应用程序,但是/bin 是可以的(例如,如果我尝试执行/bin/ls 没有问题)。

我尝试的另一个解决方案是通过设置启动路径和参数来运行 /bin/bash,如下所示:

task.launchPath = [[[NSProcessInfo processInfo] environment] objectForKey:@"SHELL"];
task.arguments = @[@"-l", @"-c", @"/usr/local/bin/clang-format -style=LLVM someFile.m"];

这成功启动了任务,但失败并显示以下错误输出:

/bin/bash: /etc/profile: 不允许操作 /bin/bash: /usr/local/bin/clang-format: 不允许操作

第一条错误消息是由于尝试在 bash 中调用 -l 参数,该参数尝试以用户身份登录。

知道如何启用对其他文件夹的访问吗?我需要启用某种沙盒环境设置吗?

4

2 回答 2

1

我想由于沙盒,这是不可能的。您可以捆绑 clang-format 可执行文件并从那里使用它。

于 2016-09-16T15:34:39.617 回答
0

就个人而言,我认为你做错了。扩展应该很快(如果你在 Xcode 扩展上观看视频,他会重复多次进出)。而且它们受到严格限制。

但是,还有另一个 - 容器应用程序可能能够为您的扩展程序执行此处理,而无需所有黑客攻击。缺点是您必须将缓冲区传入和传出扩展。

这并不容易,但可以做到。让您的容器运行的简单方法。首先,修改容器应用的 Info.plist(不是扩展名 Info.plist),使其具有 URL 类型。

信息列表

在您的扩展程序中,您可以通过运行以下命令“唤醒”容器应用程序:

let customurl = NSURL.init(string: “yoururlschemehere://")
NSWorkspace.shared().open(customurl as! URL)

至于两者之间的沟通,Apple 有很多方法。我,我是老派,所以我正在使用 DistributedNotificationCenter - 目前。

虽然我没有尝试过,但我不明白为什么容器应用程序在与 clang 聊天时会出现问题(我正在使用容器应用程序进行设置)。

于 2016-09-22T18:13:47.437 回答