CocoaPods 对 Xcode Assets 打包的诡异问题

好久没有写博客了,有一年多了吧,想想那些能够安心码字的日子还甚是怀念,于是今晚无论外界条件怎样的恶劣,这一篇是一定要更新的。

想必 CocoaPods 和 Carthage 对于 iOS 开发者而言都不会陌生,今天的这一篇我们就来看看混合使用这两者,以及多个 .xcassets 的情况下,一些莫名其妙的问题。

问题的出现

去年下半年,我们内部搭建了一个私有的 CI 平台,主要用于跑一些自动化和日常的测试打包。由于机器性能很强悍,很快成了大家构建新包的首选,原本应该是个愉快的事情,可是在系统这样跑了几个月后,恼人的问题突然就出现了:大家在自己的机器上编译没任何问题,而在 CI 里发起构建,每次都会失败在 Build Phases 中的 [CP] Copy Pods Resources 这一步,查看详细日志的话,主要是下面的这段错误:

error: None of the input catalogs contained a matching stickers icon set or app icon set named  "AppIcon".

我们的主工程比较大,代码和资源文件都很多,技术选型上是采用了 CocoaPods (1.5.x 版本) 来做模块化,模块拥有各自的 .xcassets 来存放资源,所以会有多个 .xcassets 文件。在此基础上,我们还依赖了一些 Swift 开源库,所以也使用了 Carthage 来管理依赖。

第一次解决问题

问题必须解决,但我们的 .xcassets 中,肯定是有一个含有 “AppIcon” 的,于是我找了一个时间,仔细分析了下错误的详细日志,发现里面有几处这样的警告:

warning: The app icon set name "AppIcon" is used by multiple app icon sets.

这个警告提示我们 “AppIcon” 冲突了,然后看了下冲突的路劲,尽然都是在 Carthage/Checkouts 目录下。由于我们依赖的 Carthage 库是以源码编译成 Framework 的,而这些源码中有示例和测试项目,其中包括了一些.xcassets,最主要的是 CocoaPods 把这些目录下的 .xcassets 都编译到了最终输出的目标中。

相关 issue: https://github.com/CocoaPods/CocoaPods/issues/6159#issuecomment-296698412

不查不知道,一查吓一跳啊,有种“我们 App 被偷偷植入了一些莫名其妙的资源”的感觉,也很庆幸以往的 AppIcon 能正常显示,甚至是很意外它尽然能正常显示。与此同时,错误日志中还有一些其他资源名称冲突的警告,我把这些资源名称和它对应的 .xcassets 名称一一对应的提取了出来,然后在模块间查找、对比,发现尽然存在名称一致但长相完全不同的图片,庆幸的是较新的图片得到了显示。捏了一把冷汗,在手动处理完所有名称冲突后,我把 CI 服务中的构建脚本修改了下,在编译前执行了下面命令:

rm -rf Carthage/Checkouts/**/*.xcassets || :

删除了这些不相干、也没任何作用的 .xcassets。做完这一切,我在 CI 上发起了一个构建,然后真的就成功了。可这没法解释为啥原先本地没问题,隐隐觉得还没有找到问题的主线,这次只是完成了一个支线任务。

大家又开始愉快地使用 CI 了。

第二次解决问题

时间飞快,好景不长,过了一个月左右,相同的问题、相同的错误信息再一次出现在我面前,而我再一次翻起那详细日志时,里面已没有了任何重复冲突的警告。这一次,我开始认真思考:本地发起构建和通过 CI 发起构建到执行 [CP] Copy Pods Resources 这一步,到底有什么区别?通过一系列测试,发现 Shell 的环境不同,但无法确定环境中哪些会影响到这一步执行,没办法,只能细看下 CocoaPods 这一步自动生成的脚本了:

# 其中 #{Target Name} 为主工程输出目标名称
Pods/Target Support Files/Pods-#{Target Name}/Pods-#{Target Name}-resources.sh

这个脚本主要就是拷贝和编译通过 CocoaPods 所依赖的资源文件,和 .xcassets 相关的主要是 XCASSET_FILES 这个变量,以及最后的这段脚本:

if [ -z ${ASSETCATALOG_COMPILER_APPICON_NAME+x} ]; then
    printf "%s\0" "${XCASSET_FILES[@]}" | xargs -0 xcrun actool --output-format human-readable-text --notices --warnings --platform "${PLATFORM_NAME}" --minimum-deployment-target "${!DEPLOYMENT_TARGET_SETTING_NAME}" ${TARGET_DEVICE_ARGS} --compress-pngs --compile "${BUILT_PRODUCTS_DIR}/${UNLOCALIZED_RESOURCES_FOLDER_PATH}"
else
    printf "%s\0" "${XCASSET_FILES[@]}" | xargs -0 xcrun actool --output-format human-readable-text --notices --warnings --platform "${PLATFORM_NAME}" --minimum-deployment-target "${!DEPLOYMENT_TARGET_SETTING_NAME}" ${TARGET_DEVICE_ARGS} --compress-pngs --compile "${BUILT_PRODUCTS_DIR}/${UNLOCALIZED_RESOURCES_FOLDER_PATH}" --app-icon "${ASSETCATALOG_COMPILER_APPICON_NAME}" --output-partial-info-plist "${TARGET_TEMP_DIR}/assetcatalog_generated_info_cocoapods.plist"
fi

所有的 .xcassets 文件都存在了 XCASSET_FILES 这个数组里面了,然后这个数组传递给 actool 来进行编译成 Assets.car 文件。脚本看完后,通过在文件头部加上 set -x 开启了它的调试,在茫茫的输出中,我死盯着 XCASSET_FILES 这个变量,然后惊人的发现这个变量中所有依赖而来的 .xcassets 都变成了两份!真心不知道 actool 在面对这些重复的路劲是怎么的处理,又捏了一把冷汗,于是乎在 Podfile 中加上了这样一些代码:

post_install do |installer|
   # 其中 #{Target Name} 为主工程输出目标名称
    copy_pods_resources_path = "Pods/Target Support Files/Pods-#{Target Name}/Pods-#{Target Name}-resources.sh"
    
    str1 = 'printf "%s\0" "${XCASSET_FILES[@]}"'
    str2 = 'printf "%s\n" "${XCASSET_FILES[@]}" | sort -u | tr \'\n\' \'\\\\0\' '
    
    text = File.read(copy_pods_resources_path)
    new_contents = text.gsub(str1, str2)
    File.open(copy_pods_resources_path, "w") {|file| file.puts new_contents }
end

通过对 CocoaPods 自动生成文件内容进行替换,我们插入了一段脚本,最终对 XCASSET_FILES 中的条目进行了去重。做完这一切,我又在 CI 上发起了一个构建,然后它又成功了,然后还是没法解释为啥本地没问题,所以,注定了这还是一个支线任务。

可是,大家再一次愉快地使用 CI 了。

第三次解决问题

过了很长一段时间,长到我都以为这个问题真的彻底解决了,但冷不丁的就在前几天,这个问题又出现了。都说事不过三,这问题一次又一次的反复,也实在是让我颜面尽失,大过年的,你这该死又淘气的 CocoaPods。按捺住心中的烦躁不安,我又一次仔细地把那自动生成的脚本撸了一遍,可能是内心足够安静了吧,这一次我尽然只凭理论分析,就找到了罪魁祸首 xargs

xargs 不仅能正确处理空格之类的转义,还会在超过一定的限制后,把传递给它的参数分批传递给后续的命令,XCASSET_FILES 中就是我们所存储的参数。xargs 分批传递的限制主要是两个参数:-n 的条目限制和 -s 的大小限制,其中 -s 的大小限制受环境变量 ARG_MAX 影响。

所以,一旦我们的 XCASSET_FILES 被分批传递给了 actool,其中只有某一批里面有“AppIcon”,其它的自然会报错。由于环境不同,ARG_MAX 值不一致,这也解释了一直没法解释的那个问题。前面的两次修复,都不经意间缩减了 XCASSET_FILES 中的内容,而我们的 .xcassets 文件在慢慢增多,一旦突破了限制,问题就又出现了。

感觉找打了主线,于是修改了下 Podfile 中的代码:

post_install do |installer|
   # 其中 #{Target Name} 为主工程输出目标名称
    copy_pods_resources_path = "Pods/Target Support Files/Pods-#{Target Name}/Pods-#{Target Name}-resources.sh"
    
    str1 = 'printf "%s\0" "${XCASSET_FILES[@]}" | xargs -0'
    str2 = 'printf "%s\n" "${XCASSET_FILES[@]}" | sort -u | tr \'\n\' \'\\\\0\' | xargs -0 -s 20480 -n 100'
    
    text = File.read(copy_pods_resources_path)
    new_contents = text.gsub(str1, str2)
    File.open(copy_pods_resources_path, "w") {|file| file.puts new_contents }
end

搞完后,我在 CI 上再一次发起了构建,如预期的一样,再一次的成功了,我相信这问题不会再出现了。

大家又开始愉快地使用 CI 了。

总结一下

这个问题让我纠结了大半年时间,终于在这新春佳节里给彻底解决了,大体来说就是这样:

  • 如果你混合使用了 CocoaPods 和 Carthage,确认下 Carthage 的所有目录里是否有 .xcassets,如果有的话,确认下是否被打包到你最后的 App 里了
  • 注意 CocoaPods 生成的脚本中,XCASSET_FILES 里的条目有重复
  • CocoaPods 生成的脚本中,最终传递给 actool 编译的参数,一定不能被 xargs 分批传递
  • 如果你项目中有多个 .xcassets 被 CocoaPods 管理,修改资源导致图片丢失,参考:Issue 和这个 PodResourcesBug 项目
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,794评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,050评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,587评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,861评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,901评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,898评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,832评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,617评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,077评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,349评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,483评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,199评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,824评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,442评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,632评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,474评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,393评论 2 352

推荐阅读更多精彩内容