使用fastlane实现framework一键发版

一、背景

七鱼SDK在今年经过出海业务、视频客服业务的迭代后,发布流程已经变的繁琐,这些流程不仅费时费力,还特别容易出现人为的错误。于是,我从优化打包脚本出发,实现一键自动发布版本。

二、发版流程

七鱼现有的发版流程如下图:


发版流程.png

每次发版,都需要发上图中的国内版、国际版、cocpoads版。由于这三个版本所对应的资源和代码分支是有一定差别的,所以每次发完一种版本后,都需要手动切换对应的资源和配置,切换完之后,分别打蒲公英包和SDK的包。整个流程不仅繁琐耗时,而且非常容易出错。如果这个流程某个步骤出错了,很难人为的发现,更多是依赖客户或者技术支持的同事在真正使用场景时才发现包有问题。

三、打包脚本--Fastlane

基于上述打包流程的问题,如果能够通过脚本自动去执行,那么不仅可以简化发版流程,降低错误率,而且大量节省人力。
于是,我们打算使用Fastlane来优化流程。

1、Fastlane简介

Fastlane 是一整套的客户端 CI 工具集合,替代开发者处理构建和发布 App 中繁琐的任务,可以非常快速简单的搭建一个自动化发布服务,并且支持Android,iOS,MacOS。Fastlane本身没有一套特殊语法,使用的 Ruby 语言。Fastlane的安装以及环境的搭建,网上有很多的资料可以翻阅,本文将不赘述,本文将着重介绍如何实现上述复杂流程的一键发布。

2、Fastlane的使用

搭建好Fastlane的环境以后,在工程目录下执行 fastlane init,打开Fastlane文件,就可以开始编写脚本了。

platform :ios do
  desc "Description of what the lane does"
  lane :custom_lane do
    # add actions here: https://docs.fastlane.tools/actions
  end
end

这是每一个fastlane脚本的最简单形式。最外面的"ios"层是脚本入口,类似于main函数。在脚本中可以自定义action,其中fastlane是关键字,相当于函数的function,当然,也可以写成有参的形式:

lane :custom_lane do |options|
end

fastlane提供了一些内置的action能力,可以通过查阅官方文档来获取。
fastlane中的action既可以被内部调用,也可以被外部命令行调用。
例如内部调用:

platform :ios do
  desc "Description of what the lane does"
  lane :custom_lane do
    # add actions here: https://docs.fastlane.tools/actions
  end
  lane :custom_lane_options do |options|
  custom_lane
  end
end

例如命令行调用:

bundle exec fastlane ios custom_lane

内部调用时,就和调用函数用法相同,命令行调用时,由于fastlane是依赖于bundle环境的,需要用bundle来驱动,同时需要申明执行的fastlane脚本名称“ios”。

四、一键发版

从上述发版流程我们可以知道,一次发版,总共需要产出以下几项:
1、国内版:蒲公英版本A、zip压缩包A
2、国际版:蒲公英版本B、zip压缩包B
3、cocoapods版
这三项的内容有一部分是重叠的,有一部分是不同的。
在写脚本之前,我们先对各版本内部组成有一个简单的了解:
蒲公英版本A、zip压缩包A:在线模块+视频模块+国内版NIM+国内版配置文件
蒲公英版本B、zip压缩包B:在线模块+国际版NIM+国际版配置文件
cocoapods版:在线模块+视频模块+国内版NIM(或国际版NIM)+国际版配置文件
其中,蒲公英版本是ipa包传到蒲公英平台;zip压缩包是各模块SDK组合而成。

1、各分支模块拆解

基于上述内容我们可以知道,“在线模块”是公共模块,剩余的模块是选配模块,并且各模块是需要单独打包的.其中,在线和视频模块需要使用xcframework形式的包,而NIM需要使用framework的包(保持与云信同步)。所以,我们可以先打在线模块、视频模块的包:

  lane :sdk do |options|
  #清理文件夹
    clear_derived_data(derived_data_path: "./DerivedData")
    clear_derived_data(derived_data_path: "./QYProduct")
    #打SDK包
    customSchemes = ["QYSDK", "QYVideoService"]
    for customScheme in customSchemes
      xcbuild(
        scheme: customScheme,
        configuration: "Release",
        destination: "generic/platform=iOS Simulator",
        sdk: "iphonesimulator",
        xcargs: "-quiet ARCHS='x86_64' BITCODE_GENERATION_MODE=marker ",
        derivedDataPath: "./DerivedData"
      )
      xcbuild(
          scheme: customScheme,
          destination: "generic/platform=iOS",
          configuration: "Release",
          sdk: "iphoneos",
          xcargs: "-quiet ARCHS='arm64' BITCODE_GENERATION_MODE=bitcode ",
          derivedDataPath: "./DerivedData"
      )
      create_xcframework(
        frameworks_with_dsyms: {
          "DerivedData/Build/Products/Release-iphonesimulator/#{customScheme}.framework" => { dsyms: Dir.pwd + "/../DerivedData/Build/Products/Release-iphonesimulator/#{customScheme}.framework.dSYM" },
          "DerivedData/Build/Products/Release-iphoneos/#{customScheme}.framework" => { dsyms: Dir.pwd + "/../DerivedData/Build/Products/Release-iphoneos/#{customScheme}.framework.dSYM" } },
        output: "QYProduct/QY_iOS_SDK/SDK/#{customScheme}.xcframework")
    end
  end 

解析: fastlane中支持使用Xcode的工具来打包,官方已经提供xcbuild了,各入参可以根据项目需要自行配置,分别打真机和模拟器的包即可。打完包后,需要将两种包合并成xcframework,使用create_xcframework即可。

接下来我们需要打NIM的SDK包。由于国内版和国际版使用了不同的NIM分支,因此,我们在打包之前需要先切换分支,然后再打包。然而,我们的NIM模块是以submodule的方式存在的,而fastlane本身能使用submodule的功能很有限,于是我们想到,是否可以在fastlane内部去执行git指令,这样就可以间接的完成分支切换了,于是就有如下的代码:

cmd = "git submodule foreach git checkout . && git submodule foreach git checkout 9.2.8"
`#{cmd}`

fastlane支持在控制台间接执行git指令,如果有需要连续执行多条指令,指令直接可以使用“&&”来连接。上述指令则实现了清空子模块的变更,同时将分支切换到9.2.8。仔细观察,我们会发现,指令中有一个foreach,这是循环遍历执行子模块的指令。那么问题就出现了,我的工程是有多个子模块的,而我仅仅需要切换一个模块,恰好我需要执行的子模块处于第一个,当foreach执行出错后,会停止继续执行,这样也算勉强能够实现功能。除了foreach,是否有别的命令可以实现切换单个子模块分支呢?经过一番寻找和思考后,最终还是失败了,但也有想到两个替代方案。
方案一: git是支持命令扩展的,我们可以通过扩展的方式去自定义一个切换指定子模块分支的命令,然后再通过fastlane去执行这个命令。
方案二: 用shell脚本或note.js脚本来实现切换分支,然后通过fastlane去执行脚本。
这两种方案还待后续研究,此处先粗糙使用foreach过度一下。
我们但NIM包是需要根据不同分支打不同包的,因此我们就需要使用到带参数的fastlane action,参考如下:

lane :nim_package do |options|
    case options[:type]
    when 'abroad'
    #国际版
    else
    #国内版
    end

我们可以根据入参来选择需要切换的分支,这样,我们NIM打包action的完整内容如下:

lane :nim_package do |options|
    code_path = File.expand_path("..", File.dirname(__FILE__)).to_s
    case options[:type]
    when 'abroad'
    cmd = "git submodule foreach git checkout . && git submodule foreach git checkout 9.2.8"
    clear_derived_data(derived_data_path: "./QYProduct/abroad")
    out_path = "#{code_path}/QYProduct/abroad/QY_iOS_SDK/NIMSDK"
    else
    cmd = "git submodule foreach git checkout . && git submodule foreach git checkout feature_8.9.2_noopenssl_compress"
    clear_derived_data(derived_data_path: "./QYProduct/enterprise")
    out_path = "#{code_path}/QYProduct/enterprise/QY_iOS_SDK/NIMSDK"
    end
    `#{cmd}`
    xcbuild(
        scheme: "NIMSDK",
        configuration: "Release",
        destination: "generic/platform=iOS Simulator",
        sdk: "iphonesimulator",
        xcargs: "-quiet ARCHS='x86_64' BITCODE_GENERATION_MODE=marker ",
        derivedDataPath: "./DerivedData"
      )
      xcbuild(
          scheme: "NIMSDK",
          destination: "generic/platform=iOS",
          configuration: "Release",
          sdk: "iphoneos",
          xcargs: "-quiet ARCHS='arm64' BITCODE_GENERATION_MODE=bitcode ",
          derivedDataPath: "./DerivedData"
      )

    iphonesimulator_path = "#{code_path}/DerivedData/Build/Products/Release-iphonesimulator/NIMSDK.framework"
    iphoneos_path = "#{code_path}/DerivedData/Build/Products/Release-iphoneos/NIMSDK.framework"
    command = "mkdir -p #{out_path} && cp -rf #{iphoneos_path} #{out_path}/NIMSDK.framework && lipo -create #{iphoneos_path}/NIMSDK #{iphonesimulator_path}/NIMSDK -output #{out_path}/NIMSDK.framework/NIMSDK"
    `#{command}`
  end

打完NIM的包后,我们得到了两个文件夹,分别存放了国内版和国际版的SDK,这样,我们后续发版的时候也不再需要手动调整文件夹的内容。
到目前为止,SDK的包都已经打好了,接下来需要打ipa包,并上传到蒲公英。
由于蒲公英的包也需要打国内和国外的包,同时需要分配不同的配置文件,因此,我们仍然需要用到带参数的action,并输出到指定的路径。

  lane :qy_package do |options|
    case options[:type]
    when 'abroad'
    copy_artifacts(
      target_path: 'QYDemo/Resources',
      artifacts: ['Abroad/QYConfigResource.bundle']
    )
    abroad
    copy_artifacts(
      target_path: 'QYProduct/abroad/QY_iOS_SDK/SDK',
      artifacts: ['QYProduct/QY_iOS_SDK/SDK/QYSDK.xcframework']
    )
    else
    copy_artifacts(
      target_path: 'QYDemo/Resources',
      artifacts: ['Default/QYConfigResource.bundle']
    )
    enterprise
    copy_artifacts(
      target_path: 'QYProduct/enterprise/QY_iOS_SDK/SDK',
      artifacts: ['QYProduct/QY_iOS_SDK/SDK/QYSDK.xcframework','QYProduct/QY_iOS_SDK/SDK/QYVideoService.xcframework']
    )
    end
  end

这里涉及到打包上传到蒲公英的环节,这里面涉及到证书的管理,可以参考另一篇文章fastlane通过match管理证书。我们先看一下打包并上传蒲公英的action:

platform :ios do
  desc "企业包"
  lane :enterprise do |options|
    sync_code_signing(
      type: "enterprise",
      app_identifier: ["bundleId"],
      readonly: true,
      username: "用户名",
      keychain_password: options[:keychain_password])
    build_app(
      scheme: "QYDemo",
      include_bitcode: false,
      destination: "generic/platform=iOS",
    )
    pgyer(api_key: "api_key",
      password: "蒲公英包下载密码",
      install_type: "2",
      update_description: "update by fastlane")
  end

涉及的action内容,在fastlane通过match管理证书有详细讲解,此文不重复赘述。

2、完整打包及优化风险项

至此,我们已经将所有需要打的包都打好了,接下来需要对一下额外的资源文件进一步分配处理,同时将调用上述所有的action:

lane :sdk_package do |options|
    clear_derived_data(derived_data_path: "./DerivedData")
    clear_derived_data(derived_data_path: "./QYProduct")
    #打SDK包
    sdk
    #国内
    nim_package(type:'enterprise')
    qy_package(type:'enterprise')
    #国外
    nim_package(type:'abroad')
    qy_package(type:'abroad')

    
    version = get_version_number(
    xcodeproj: "QYDemo.xcodeproj",
    target: "QYDemo"
    )
    version_bump_podspec(
    path: "QY_iOS_SDK.podspec", 
    version_number: "#{version}"
    )
    copy_artifacts(
      target_path: 'QYProduct/QY_iOS_SDK',
      artifacts: ['QY_iOS_SDK.podspec']
    )
    targets = ["abroad/QY_iOS_SDK","enterprise/QY_iOS_SDK","QY_iOS_SDK/SDK"]
    for target in targets
      copy_artifacts(
      target_path: "QYProduct/#{target}",
      artifacts: ['开发指南.md','Abroad/资源文件说明.md']
      )
      copy_artifacts(
      target_path: "QYProduct/#{target}/Resources",
      artifacts: ['QYDemo/Resources/QYResource.bundle', 'QYDemo/Resources/QYCustomResource.bundle', 'QYDemo/Resources/QYLanguage.bundle']
      )
      if target != "enterprise/QY_iOS_SDK"
    copy_artifacts(
      target_path: "QYProduct/#{target}/Resources",
      artifacts: ['Abroad/QYConfigResource.bundle']
      )
      end
      if target != "abroad/QY_iOS_SDK"
    copy_artifacts(
      target_path: "QYProduct/#{target}/Resources",
      artifacts: ['QYDemo/Resources/QYVideoResource.bundle']
      )
      end
      if target == "abroad/QY_iOS_SDK"
      zip(
      path: "QYProduct/#{target}",
      output_path: "QYProduct/QY_iOS_SDK_v#{version}_Abroad.zip"
      )
      end
      if target == "enterprise/QY_iOS_SDK"
      zip(
      path: "QYProduct/#{target}",
      output_path: "QYProduct/QY_iOS_SDK_v#{version}.zip"
      )
      end
    end
    #打包完,恢复对应分支
    cmd = "git submodule foreach git checkout . && git submodule foreach git checkout feature_8.9.2_noopenssl_compress"
   `#{cmd}`
    copy_artifacts(
      target_path: 'QYDemo/Resources',
      artifacts: ['Default/QYConfigResource.bundle']
    )
    sentry_upload_dif(
      path: Dir.pwd + '/../QYProduct/'
    )
  end

这是一个主action,通过嵌套调用子action实现打包,在打完包后,将不同的资源和文件分配到每个包的路径中,同时,读取好版本号后写入podspec文件。
至此,我们完整的打包脚本已经完成,我们重新梳理一下这个脚本它做了哪些事情:
1、打在线模块、视频模块的xcframework
2、切换分支,并打不同分支的NIM的framework
3、根据不同的分支,打不同的ipa包,并使用match来自动管理证书,同时将包上传蒲公英
4、更新podspec、readme、资源包等内容,并分发到各版本对应文件夹路径
5、打zip压缩包,将分支恢复初始状态。
经过上述5个环节后,我们最终会得到如下的产物:


image.png
image.png

接下来,我们仅需将对应的内容,发布到官网、cocoapods即可。
继续优化方向:
1、优化foreach遍历所有子模块的方式,实现精准切换子模块
风险考虑:
由于发版时需要将打包产物上传运维官网后台和cocoapods仓库,因此,可以让服务端提供自动上传的接口,通过脚本自动传包。
方向2实现后,整个发版过程仅需要一行命令即可完成,但由于每次打完包都是自动上传发布,没办法人为检验正确性,如果打包出了问题,或者打错包了,也会直接发出去,而cocoapods的版本管理是需要升版本号的,一旦出错,旧的版本号就废弃不能用了,存在一定的风险。

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

推荐阅读更多精彩内容