Xcode 10:由于无法签署第 3 方依赖框架 (PromiseKit),因此对我的 App+Framework 进行代码签名失败。在 Xcode 9 工作

Xcode 10: Code Signing my App+Framework fails, because of failure signing 3rd party dependency framework (PromiseKit). Works in Xcode 9

我有一个 Xcode 10 - iOS12 swift 项目链接到我自己的框架(还有 Xcode 10 + iOS12)。

应用项目正在引用我的框架项目作为子项目引用。

我的框架项目引用 PromiseKit.framework(通用框架 - 胖库),使用以下构建脚本制作:

# Merge Script

# 1
# Set bash script to exit immediately if any commands fail.
set -e

# 2
# Setup some constants for use later on.
FRAMEWORK_NAME="PromiseKit"

# 3
# If remnants from a previous build exist, delete them.
if [ -d "${SRCROOT}/build" ]; then
rm -rf "${SRCROOT}/build"
fi

# 4
# Build the framework for device and for simulator (using
# all needed architectures).
xcodebuild -target "${FRAMEWORK_NAME}" -configuration Release -arch arm64 only_active_arch=no defines_module=yes -sdk "iphoneos"
xcodebuild -target "${FRAMEWORK_NAME}" -configuration Release -arch x86_64 only_active_arch=no defines_module=yes -sdk "iphonesimulator"

# 5
# Remove .framework file if exists on Desktop from previous run.
if [ -d "${SRCROOT}/${FRAMEWORK_NAME}.framework" ]; then
rm -rf "${SRCROOT}/${FRAMEWORK_NAME}.framework"
fi

# 6
# Copy the device version of framework to Desktop.
cp -r "${SRCROOT}/build/Release-iphoneos/${FRAMEWORK_NAME}.framework" "${SRCROOT}/${FRAMEWORK_NAME}.framework"

# 7
# Replace the framework executable within the framework with
# a new version created by merging the device and simulator
# frameworks' executables with lipo.
lipo -create -output "${SRCROOT}/${FRAMEWORK_NAME}.framework/${FRAMEWORK_NAME}" "${SRCROOT}/build/Release-iphoneos/${FRAMEWORK_NAME}.framework/${FRAMEWORK_NAME}" "${SRCROOT}/build/Release-iphonesimulator/${FRAMEWORK_NAME}.framework/${FRAMEWORK_NAME}"

# 8
# Copy the Swift module mappings for the simulator into the
# framework.  The device mappings already exist from step 6.
cp -r "${SRCROOT}/build/Release-iphonesimulator/${FRAMEWORK_NAME}.framework/Modules/${FRAMEWORK_NAME}.swiftmodule/" "${SRCROOT}/${FRAMEWORK_NAME}.framework/Modules/${FRAMEWORK_NAME}.swiftmodule"

# 9
# Delete the most recent build.
if [ -d "${SRCROOT}/build" ]; then
rm -rf "${SRCROOT}/build"
fi

当我使用 Xcode 10(以及 9.4.1)和 Bitcode ON(包含我的框架参考和 PromiseKit Fat 库)来存档我的父应用程序项目时,我收到以下错误签名阶段:(无法验证 PromiseKit 中的位码。framework/PromiseKit:错误无法从 /var/folders.../(x86_64) 中提取包 - 这表明它与模拟器切片相关)

如果我将管理器中的 "Rebuild from Bitcode" 选项关闭,则会出现不同的错误:(代码签名 "PromiseKit.framework" 失败)

但是,如果我使用 Xcode 9.4.1 并关闭位码,则它可以正常导出和签名。

为什么要尝试单独重新签署子框架,我可以做些什么来缓解这些问题?我需要归档才能正常使用 Xcode 10,以及将任何未来的第三方依赖项添加到我的框架目标中。 (这是添加到我的 Framework 目标的第一个动态框架依赖项。在我 "baking in" 之前 - 加入所有第 3 方以方便开发,但由于对 [=42= 的广泛依赖,PromiseKit 很难加入]).

Xcode 存档日志是:

 {
        code = 330;
        description = "Failed to resolve linkage dependency PromiseKit x86_64 -> @rpath/libswiftFoundation.dylib: Unknown arch x86_64";
        info =             {
        };
        level = WARN;
    },
            {
        code = 330;
        description = "Failed to resolve linkage dependency PromiseKit x86_64 -> @rpath/libswiftObjectiveC.dylib: Unknown arch x86_64";
        info =             {
        };
        level = WARN;
    },
            {
        code = 0;
        description = "Failed to verify bitcode in PromiseKit.framework/PromiseKit:\nerror: Cannot extract bundle from /var/folders/q5/hm9v_6x53lj0gj02yxqtkmd40000gn/T/IDEDistributionOptionThinning.RJD/Payload/MyAppName.app/Frameworks/PromiseKit.framework/PromiseKit (x86_64)\n\n";
        info =             {
        };
        level = ERROR;
        type = "malformed-payload";
    }
);

我尝试的其他一些解决方案是使用对 PromiseKit 的项目引用,而不是框架引用,但这不起作用 - 因为我仍然需要来自我的主项目的框架引用,因为我会得到 "library not loaded" 运行时错误,如果 运行 没有 FW 引用。使用项目引用进行归档时会出现同样的问题。

同样的问题。我发现的唯一解决方法是使用静态库而不是框架。

如果您无法使用静态库,您最好向 Apple 提交错误报告。

试试这个!它对我和许多其他人都有效:

转到

构建阶段 > 添加 > 新建 运行 脚本阶段

代码应该适用于任何默认值 shell,但我建议只使用 /bin/sh

并包含以下代码:

# Type a script or drag a script file from your workspace to insert its path.
# skip if we run in debug
if [ "$CONFIGURATION" == "Debug" ]; then
echo "Skip frameworks cleaning in debug version"
exit 0
fi

APP_PATH="${TARGET_BUILD_DIR}/${WRAPPER_NAME}"

# This script loops through the frameworks embedded in the application and
# removes unused architectures.
find "$APP_PATH" -name '*.framework' -type d | while read -r FRAMEWORK
do
FRAMEWORK_EXECUTABLE_NAME=$(defaults read "$FRAMEWORK/Info.plist" CFBundleExecutable)
FRAMEWORK_EXECUTABLE_PATH="$FRAMEWORK/$FRAMEWORK_EXECUTABLE_NAME"
echo "Executable is $FRAMEWORK_EXECUTABLE_PATH"

EXTRACTED_ARCHS=()

for ARCH in $ARCHS
do
echo "Extracting $ARCH from $FRAMEWORK_EXECUTABLE_NAME"
lipo -extract "$ARCH" "$FRAMEWORK_EXECUTABLE_PATH" -o "$FRAMEWORK_EXECUTABLE_PATH-$ARCH"
EXTRACTED_ARCHS+=("$FRAMEWORK_EXECUTABLE_PATH-$ARCH")
done

echo "Merging extracted architectures: ${ARCHS}"
lipo -o "$FRAMEWORK_EXECUTABLE_PATH-merged" -create "${EXTRACTED_ARCHS[@]}"
rm "${EXTRACTED_ARCHS[@]}"

echo "Replacing original executable with thinned version"
rm "$FRAMEWORK_EXECUTABLE_PATH"
mv "$FRAMEWORK_EXECUTABLE_PATH-merged" "$FRAMEWORK_EXECUTABLE_PATH"

done

似乎有些框架发布了架构,但不会在应用程序中使用。 Xcode 将拒绝签署。上面的脚本删除了未使用的架构。

致谢:GitHub 的某个人,我找不到确切的来源了。

这对我有用, 我有应用程序和 2 个内部构建的框架,比如 A $ B.

应用程序需要 A,但 A 需要 B,并且由于 Apple 不推荐嵌套框架,因此 A 和 B 都必须包含在应用程序中。

这就是我的 Xcode 项目的样子。

解决方案

在应用程序中,在 Frameworks、Libraries 和 Embedded Content、select Embed & Sign 下获取所有必要的框架。 (如下图)

但对于所有自定义框架项目,在 Frameworks and Libraries 部分下,select Do Not Embed。 (如下所示) 这解决了我的问题