带有不安全构建标志的 Swift 包依赖项:目标完整性错误

Posted

技术标签:

【中文标题】带有不安全构建标志的 Swift 包依赖项:目标完整性错误【英文标题】:Swift package dependency with unsafe build flags: target integrity error 【发布时间】:2020-11-13 18:20:44 【问题描述】:

TL;DR

尝试在 Xcode 中的 ios 项目中使用本地 Swift 包依赖项时,我收到以下错误:

The package product 'DlibWrapper' cannot be used as a dependency of this target because it uses unsafe build flags.

(我使用不安全标志来指定静态库位置)

我也尝试将包导入为基于远程分支的依赖项,但它还是失败了。

根据 Swift 论坛上的this post,该问题已在不久前得到解决,相应的pull request 已被合并。

问题出现在 Swift 5.2.4 (Xcode 11.6) 和 5.3 (Xcode 12 beta 3) 中。

任何线索可能是什么问题?


详情

我正在尝试构建一个包装 dlib 库的 Swift 包,并在 iOS 应用程序中使用它。由于平台的原因,我不能使用.systemLibrary 目标来链接dlib。所以我将它预编译在一个静态库中,并与包装器代码一起打包,如下所示:

DlibWrapper/
  Libraries/
    dlib/
      include/
          ...
      lib/
        arm64/
          libdlib.a

  Sources/
    CWrapper/
      include/
        module.modulemap
        cwrapper.h
      cwrapper.cpp

    SwiftWrapper/
      SwiftWrapper.swift

  Package.swift

DlibWrapper/Package.swift的简化内容:

// swift-tools-version:5.3

import PackageDescription

let package = Package(
    name: "DlibWrapper",
    platforms: [
        .iOS(.v13)
    ],
    products: [
        .library(
            name: "DlibWrapper",
            targets: ["CWrapper", "SwiftWrapper"])
    ],
    dependencies: [],
    targets: [
        .target(
            name: "SwiftWrapper",
            dependencies: ["CWrapper"]
        ),
        .target(
            name: "CWrapper",

            cxxSettings: [.headerSearchPath("../../Libraries/dlib/include")],

            linkerSettings: [
                .linkedLibrary("dlib"),
                .linkedFramework("Accelerate", .when(platforms: [.iOS])),

                // The error is caused by this line
                .unsafeFlags(["-LLibraries/dlib/lib/arm64"], .when(platforms: [.iOS])),
            ]
        ),
    ],
    cxxLanguageStandard: .cxx1z
)

我尝试在module.modulemap 中使用link 属性,但编译器似乎忽略了它。此外,在目标清单中的 .linkedLibrary() 中提供库的绝对路径也无济于事,链接器会抱怨它找不到库。

任何解决方法的想法? (作为最后的手段,我可​​能会将所有内容打包在一个框架中)

不胜感激。

谢谢

【问题讨论】:

我也遇到了这个问题。我的解决方案是停止尝试构建第三方库的 Swift 包。我将尝试为他们构建 Xcode 项目。 【参考方案1】:

更新

原来下面的方法实际上不起作用。当包被编译为静态库时,Swift 完全忽略了.linkedLibrary(检查swift build --verbose)。该设置仅用于动态库。然而,: 语法在后一种情况下没有任何作用——无论如何都找不到库。

(由于历史原因保留以下答案,也许它可以帮助某人找到解决方案)


旧答案

解决方法是在链接器选项中使用 clang/gcc 冒号语法,即-l:(减去-el-冒号)。详情请见this answer。

这是包清单中的相关 sn-p:

// swift-tools-version:5.3

import PackageDescription

let package = Package(
    name: "DlibWrapper",
    //...
    targets: [
        //...
        .target(
            name: "CWrapper",

            cxxSettings: [.headerSearchPath("../../Libraries/dlib/include")],

            linkerSettings: [
                // Note the `:` in front of the relative path
                .linkedLibrary(":Libraries/dlib/lib/arm64/libdlib.a", .when(platforms: [.iOS])),
                //...
            ]
        ),
    ],
    //...
)

说明

.linkedLibrary 链接器设置必不可少,为链接器提供了-l 标志。所以,你放在那里的任何参数都会被翻译成这样的东西

swift build -Xlinker -l<your-linkedLibrary-string>

: 语法允许指定具有非规范名称的库,即实际文件名。最后,由于链接器的工作目录设置为包根目录,我们可以为其提供库的相对路径。

我仍然认为这是一种解决方法,因为正确的方法是使用librarySearchPath 之类的东西,即-L 链接器选项,类似于cxxSettings 中的headerSearchPath

【讨论】:

Xcode 13 允许您在被分支名称引用时使用带有不安全标志的模块。 我昨天遇到了类似的问题:github.com/Subito-it/SBTUITestTunnel/issues/133

以上是关于带有不安全构建标志的 Swift 包依赖项:目标完整性错误的主要内容,如果未能解决你的问题,请参考以下文章

解包依赖项需要很长时间

如何在 Xcode 11 中删除 Swift 包依赖项?

使用 Swift 包管理器抑制来自依赖项的警告

如何为蒸汽添加新的依赖项?

maven2 多项目构建中缺少 aar 文件

使用 Swift 5.1 将另一个包作为 SPM 中的依赖项导入另一个包