【iOS】分享一個ipa打包腳本

寫在前面

之前寫了一個很簡單的ipa打包腳本,主要是用Xcode命令中的xcodebuild和xcrun命令來完成的,其中打包ipa的命令用了PackageApplication蛤育,本來用的好好的,升級了Xcode9之后葫松,發(fā)現(xiàn)蘋果把PackageApplication這個東西給刪了瓦糕,于是腳本就跪了。腋么。咕娄。
所以,這兩個月我用了一個很原始的方法來打ipa包珊擂。大家也可以試試(大家常用的直接用Xcode中的Product->Archive的方法我就懶得說了圣勒,大家都懂):
1.自己在Xcode配置好項目的簽名
2.用Generic iOS Device來build工程
3.找到生成的.app包,然后新建一個名為Payload的文件夾摧扇,將.app包放到文件夾里面
4.壓縮Payload文件夾圣贸,然后將壓縮包的后綴從.zip改成.ipa。
然后就可以將ipa包安裝到手機(jī)上測試了扛稽。


用Generic iOS Device來build工程

壓縮Payload文件夾吁峻,然后將壓縮包的后綴從.zip改成.ipa

為什么需要一個ipa打包腳本

嗯,因為想偷懶。
直接在Xcode上點擊Product->Archive->此處省略n步選擇.......這種方式確實是挺方便的用含,估計很多人也是這么干的矮慕。但是,當(dāng)你多次打包了之后你就發(fā)現(xiàn)啄骇,打包這事好無聊痴鳄。每次都是同樣的配置,同樣的操作步驟缸夹,同樣的選擇夏跷,這么簡單的操作我居然要重復(fù)N遍。明未。槽华。而且有時候要等很久才能進(jìn)行下一步√送祝總之我覺得這種方式操作多了很是蛋疼猫态。
我一直秉承一個理念:能用機(jī)器自動解決的問題盡量不用人工操作
要是有一個東西披摄,只要我配置好了一次之后亲雪,以后直接雙擊就能直接打包,是不是比之前的方式更好疚膊?
嗯义辕,有挺多這種工具的,比如shenzhen寓盗,fastlane等等灌砖。
不過我是自己寫了一個簡單的shell腳本來實現(xiàn)功能的,比較簡單傀蚌,而且不會對項目有侵入性基显。借此還可以順便學(xué)習(xí)下shell腳本使用。

自動打包腳本如何使用善炫?

腳本的github地址為:https://github.com/shixueqian/AutoPackageScript
使用方法:

  • 1.將腳本復(fù)制到工程的根目錄
  • 2.用代碼編輯軟件(比如Xcode)打開腳本撩幽,然后根據(jù)情況修改腳本內(nèi)的一些參數(shù)
  • 3.打開終端,輸入 sh ${打包腳本的全路徑}就可執(zhí)行打包腳本箩艺。
    比如我的項目工程在
    /Users/mac/Desktop/AutoPackageScriptDemo,
    那么我的腳本路徑應(yīng)該是
    /Users/mac/Desktop/AutoPackageScriptDemo/AutoPackageScript.sh,
    所以我要執(zhí)行的命令是
    sh /Users/mac/Desktop/AutoPackageScriptDemo/AutoPackageScript.sh

sh ${打包腳本的全路徑}這行命令的作用是執(zhí)行shell打包腳本窜醉。
除了這樣執(zhí)行,我們還可以直接雙擊腳本文件就執(zhí)行腳本艺谆,不過在這之前我們需要進(jìn)行一些設(shè)置榨惰。
首先,將AutoPackageScript.sh文件的擴(kuò)展名去掉擂涛,變成AutoPackageScript
然后读串,打開終端聊记,執(zhí)行命令 chmod +x ${打包腳本的全路徑},這樣可以給腳本加上可執(zhí)行權(quán)限,并且默認(rèn)的打開方式是終端恢暖。
例如:chmod +x /Users/mac/Desktop/AutoPackageScriptDemo/AutoPackageScript
以后排监,直接鼠標(biāo)雙擊就可以執(zhí)行腳本了

腳本參數(shù)配置

看了腳本的使用,其中有一個很關(guān)鍵的東西杰捂,腳本參數(shù)配置舆床。我對多種情況都有適配,所以腳本參數(shù)會稍微有點復(fù)雜嫁佳,以下我們慢慢道來挨队。

項目的工程結(jié)構(gòu)

我也不知道怎么稱呼這個東西,簡單來講就是你的工程里面是不是使用xcworkspace(工作空間)來管理你的工程蒿往。
我們在Xcode直接新建一個iOS工程盛垦,這個時候僅僅只有一個.xcodeproj文件,是沒有.xcworkspace文件的瓤漏。
而很多的項目都采用了cocoapods來管理項目腾夯,這個時候是有.xcworkspace文件的,cocoapods通過xcworkspace來管理了第三方庫蔬充。
這兩種結(jié)構(gòu)的參數(shù)配置是不一樣的蝶俱。(因為寫腳本的時候有區(qū)別)

# 是否編譯工作空間 (例:若是用Cocopods管理的.xcworkspace項目,賦值true;用Xcode默認(rèn)創(chuàng)建的.xcodeproj,賦值false)
is_workspace="false"

# .xcworkspace的名字,如果is_workspace為true饥漫,則必須填榨呆。否則可不填
workspace_name=""

# .xcodeproj的名字,如果is_workspace為false庸队,則必須填积蜻。否則可不填
project_name="AutoPackageScriptDemo"

# 指定項目的scheme名稱(也就是工程的target名稱),必填
scheme_name="AutoPackageScriptDemo"

注釋其實已經(jīng)寫得很清楚了皿哨。舉個例子浅侨,github上的workspace_demo是通過.xcworkspace來管理的:


通過.xcworkspace來管理

項目的scheme_name為AutoPackageScriptDemo

所以這幾個參數(shù)就是

is_workspace="true"
workspace_name="AutoPackageScriptDemo"
project_name=""
scheme_name="AutoPackageScriptDemo"

否則纽谒,像github上面的project_demo里面的單工程結(jié)構(gòu)就是

is_workspace="false"
workspace_name=""
project_name="AutoPackageScriptDemo"
scheme_name="AutoPackageScriptDemo"

method,打包的方式鼓黔。

證書簽名的方式央勒,是通過腳本中的method變量控制的。
分別為 development, ad-hoc, app-store, enterprise 澳化〈薏剑看到這幾個參數(shù)估計都明白了吧?還不是不太了解的話建議看下蒲公英的這篇文檔

# method缎谷,打包的方式井濒。方式分別為 development, ad-hoc, app-store, enterprise 。必填
method="development"

profile文件的管理方式

一般來說,證書管理方式瑞你,如今應(yīng)該挺多人使用Xcode自動管理的酪惭。省心而且方便(老實說,由于工作原因者甲,我平常比較少用自動管理春感,都是手動管理的,所以理解有誤的話請?zhí)岢觯?br> 在Xcode->Preferrence->Account里面添加開發(fā)者賬號虏缸,然后在工程的General->勾選Automatically manage signing->選擇開發(fā)者賬號鲫懒。就可以自動管理了。
另外一種就是古老的手動管理方式了刽辙。
在開發(fā)者后臺上面創(chuàng)建BundleID窥岩,然后創(chuàng)建mobileprovision文件,安裝到Xcode上面選擇使用宰缤。
針對這兩種方式有不同的配置谦秧。

#  下面兩個參數(shù)只是在手動指定Pofile文件的時候用到,如果使用Xcode自動管理Profile,直接留空就好
# (跟method對應(yīng)的)mobileprovision文件名撵溃,需要先雙擊安裝.mobileprovision文件.手動管理Profile時必填
mobileprovision_name=""

# 項目的bundleID疚鲤,手動管理Profile時必填
bundle_identifier=""

注釋講得很清楚,使用Xcode自動管理profile的話直接留空就好了缘挑。
如果使用手動管理的話就需要填寫對應(yīng)的參數(shù)了集歇。

源碼解析

腳本代碼其實很簡單的。主要講解一下關(guān)鍵代碼语淘。
先把源碼放出來:

#!/bin/sh

# 使用方法:
# step1: 將該腳本放在工程的根目錄下(跟.xcworkspace文件or .xcodeproj文件同目錄)
# step2: 根據(jù)情況修改下面的參數(shù)
# step3: 打開終端诲宇,執(zhí)行腳本。(輸入sh 惶翻,然后將腳本文件拉到終端姑蓝,會生成文件路徑,然后enter就可)

# =============項目自定義部分(自定義好下列參數(shù)后再執(zhí)行該腳本)=================== #

# 是否編譯工作空間 (例:若是用Cocopods管理的.xcworkspace項目,賦值true;用Xcode默認(rèn)創(chuàng)建的.xcodeproj,賦值false)
is_workspace="false"

# .xcworkspace的名字吕粗,如果is_workspace為true纺荧,則必須填。否則可不填
workspace_name=""

# .xcodeproj的名字颅筋,如果is_workspace為false宙暇,則必須填。否則可不填
project_name="AutoPackageScriptDemo"

# 指定項目的scheme名稱(也就是工程的target名稱)议泵,必填
scheme_name="AutoPackageScriptDemo"

# 指定要打包編譯的方式 : Release,Debug占贫。一般用Release。必填
build_configuration="Release"

# method先口,打包的方式型奥。方式分別為 development, ad-hoc, app-store, enterprise 瞳收。必填
method="development"


#  下面兩個參數(shù)只是在手動指定Pofile文件的時候用到,如果使用Xcode自動管理Profile,直接留空就好
# (跟method對應(yīng)的)mobileprovision文件名厢汹,需要先雙擊安裝.mobileprovision文件.手動管理Profile時必填
mobileprovision_name=""

# 項目的bundleID缎讼,手動管理Profile時必填
bundle_identifier=""


echo "--------------------腳本配置參數(shù)檢查--------------------"
echo "\033[33;1mis_workspace=${is_workspace} "
echo "workspace_name=${workspace_name}"
echo "project_name=${project_name}"
echo "scheme_name=${scheme_name}"
echo "build_configuration=${build_configuration}"
echo "bundle_identifier=${bundle_identifier}"
echo "method=${method}"
echo "mobileprovision_name=${mobileprovision_name} \033[0m"


# =======================腳本的一些固定參數(shù)定義(無特殊情況不用修改)====================== #

# 獲取當(dāng)前腳本所在目錄
script_dir="$( cd "$( dirname "$0"  )" && pwd  )"
# 工程根目錄
project_dir=$script_dir

# 時間
DATE=`date '+%Y%m%d_%H%M%S'`
# 指定輸出導(dǎo)出文件夾路徑
export_path="$project_dir/Package/$scheme_name-$DATE"
# 指定輸出歸檔文件路徑
export_archive_path="$export_path/$scheme_name.xcarchive"
# 指定輸出ipa文件夾路徑
export_ipa_path="$export_path"
# 指定輸出ipa名稱
ipa_name="${scheme_name}_${DATE}"
# 指定導(dǎo)出ipa包需要用到的plist配置文件的路徑
export_options_plist_path="$project_dir/ExportOptions.plist"


echo "--------------------腳本固定參數(shù)檢查--------------------"
echo "\033[33;1mproject_dir=${project_dir}"
echo "DATE=${DATE}"
echo "export_path=${export_path}"
echo "export_archive_path=${export_archive_path}"
echo "export_ipa_path=${export_ipa_path}"
echo "export_options_plist_path=${export_options_plist_path}"
echo "ipa_name=${ipa_name} \033[0m"

# =======================自動打包部分(無特殊情況不用修改)====================== #

echo "------------------------------------------------------"
echo "\033[32m開始構(gòu)建項目  \033[0m"
# 進(jìn)入項目工程目錄
cd ${project_dir}

# 指定輸出文件目錄不存在則創(chuàng)建
if [ -d "$export_path" ] ; then
    echo $export_path
else
    mkdir -pv $export_path
fi

# 判斷編譯的項目類型是workspace還是project
if $is_workspace ; then
# 編譯前清理工程
xcodebuild clean -workspace ${workspace_name}.xcworkspace \
                 -scheme ${scheme_name} \
                 -configuration ${build_configuration}

xcodebuild archive -workspace ${workspace_name}.xcworkspace \
                   -scheme ${scheme_name} \
                   -configuration ${build_configuration} \
                   -archivePath ${export_archive_path}
else
# 編譯前清理工程
xcodebuild clean -project ${project_name}.xcodeproj \
                 -scheme ${scheme_name} \
                 -configuration ${build_configuration}

xcodebuild archive -project ${project_name}.xcodeproj \
                   -scheme ${scheme_name} \
                   -configuration ${build_configuration} \
                   -archivePath ${export_archive_path}
fi

#  檢查是否構(gòu)建成功
#  xcarchive 實際是一個文件夾不是一個文件所以使用 -d 判斷
if [ -d "$export_archive_path" ] ; then
    echo "\033[32;1m項目構(gòu)建成功 ?? ?? ??  \033[0m"
else
    echo "\033[31;1m項目構(gòu)建失敗 ?? ?? ??  \033[0m"
    exit 1
fi
echo "------------------------------------------------------"

echo "\033[32m開始導(dǎo)出ipa文件 \033[0m"


# 先刪除export_options_plist文件
if [ -f "$export_options_plist_path" ] ; then
    #echo "${export_options_plist_path}文件存在,進(jìn)行刪除"
    rm -f $export_options_plist_path
fi
# 根據(jù)參數(shù)生成export_options_plist文件
/usr/libexec/PlistBuddy -c  "Add :method String ${method}"  $export_options_plist_path
/usr/libexec/PlistBuddy -c  "Add :provisioningProfiles:"  $export_options_plist_path
/usr/libexec/PlistBuddy -c  "Add :provisioningProfiles:${bundle_identifier} String ${mobileprovision_name}"  $export_options_plist_path


xcodebuild  -exportArchive \
            -archivePath ${export_archive_path} \
            -exportPath ${export_ipa_path} \
            -exportOptionsPlist ${export_options_plist_path} \
            -allowProvisioningUpdates

# 檢查ipa文件是否存在
if [ -f "$export_ipa_path/$scheme_name.ipa" ] ; then
    echo "\033[32;1mexportArchive ipa包成功,準(zhǔn)備進(jìn)行重命名\033[0m"
else
    echo "\033[31;1mexportArchive ipa包失敗 ?? ?? ??     \033[0m"
    exit 1
fi

# 修改ipa文件名稱
mv $export_ipa_path/$scheme_name.ipa $export_ipa_path/$ipa_name.ipa

# 檢查文件是否存在
if [ -f "$export_ipa_path/$ipa_name.ipa" ] ; then
    echo "\033[32;1m導(dǎo)出 ${ipa_name}.ipa 包成功 ??  ??  ??   \033[0m"
    open $export_path
else
    echo "\033[31;1m導(dǎo)出 ${ipa_name}.ipa 包失敗 ?? ?? ??     \033[0m"
    exit 1
fi

# 刪除export_options_plist文件(中間文件)
if [ -f "$export_options_plist_path" ] ; then
    #echo "${export_options_plist_path}文件存在坑匠,準(zhǔn)備刪除"
    rm -f $export_options_plist_path
fi

# 輸出打包總用時
echo "\033[36;1m使用AutoPackageScript打包總用時: ${SECONDS}s \033[0m"

exit 0

clean工程

以單工程項目為例(wordspace結(jié)構(gòu)是相似的)血崭,參數(shù)配置完成之后,clean下工程厘灼,用以清除緩存夹纫,保證我們的項目是純凈的。
跟我們在Xcode中Product->Clean這個命令是一樣的功能设凹。

xcodebuild clean -project ${project_name}.xcodeproj \
                 -scheme ${scheme_name} \
                 -configuration ${build_configuration}

project_name就是我們之前配置的工程的名稱舰讹,
scheme_name就是之前配置的工程的target名稱
build_configuration為之前配置的設(shè)置,Debug或者Release
以我們demo中的單工程為例闪朱,就是這樣的

xcodebuild clean -project  AutoPackageScriptDemo.xcodeproj -scheme AutoPackageScriptDemo -configuration Release

archive工程

build工程月匣,然后archive到一個文件夾里面。

xcodebuild archive -project ${project_name}.xcodeproj \
                   -scheme ${scheme_name} \
                   -configuration ${build_configuration} \
                   -archivePath ${export_archive_path}

其中奋姿, export_archive_path就是我們存放archive結(jié)果的文件夾锄开。
這條命令執(zhí)行以后,會在指定的位置生成.xcarchive文件(其實這個是文件夾來的称诗。里面會有.app包和dSYM符號文件等內(nèi)容)
以我們的單工程demo為例:

export_archive_path=/Users/mac/Desktop/簡書/AutoPackageScript/project_demo/AutoPackageScriptDemo/Package/AutoPackageScriptDemo_20180108/AutoPackageScriptDemo.xcarchive
xcodebuild archive -project AutoPackageScriptDemo.xcodeproj -scheme AutoPackageScriptDemo -configuration Release -archivePath ${export_archive_path}
命令執(zhí)行成功會生成.xcarchive文件

.xcarchive文件里面的內(nèi)容

exportArchive,從xcarchive中導(dǎo)出ipa包

將剛才archive出來的.xcarchive文件萍悴,導(dǎo)出成一個ipa包。

xcodebuild  -exportArchive \
            -archivePath ${export_archive_path} \
            -exportPath ${export_ipa_path} \
            -exportOptionsPlist ${export_options_plist_path} \
            -allowProvisioningUpdates

export_archive_path就是剛才導(dǎo)出.xcarchive文件路徑
export_ipa_path是要導(dǎo)出的ipa文件的文件夾路徑
export_options_plist_path這是一個plist配置文件路徑寓免。這個plist配置文件是必須的癣诱,不加的話會報錯的。經(jīng)過我的測試發(fā)現(xiàn)袜香,主要是需要配置method這個key撕予。如果是手動管理的話,還需要配置bundleID和mobileprovision_name蜈首。其他的key是可以不配置的实抡,如果不配置的話,會自動根據(jù)項目里面的配置進(jìn)行生成疾就。另外澜术,這個文件我在腳本中用PlistBuddy命令直接生成了,不需要用戶自己指定文件猬腰。
-allowProvisioningUpdates主要目的是自動更新profile文件,在Xcode自動管理profile的時候用猜敢。
以我們的單工程demo為例:

export_ipa_path=/Users/mac/Desktop/簡書/AutoPackageScript/project_demo/AutoPackageScriptDemo/Package/AutoPackageScriptDemo_20180108
export_options_plist_path=/Users/mac/Desktop/簡書/AutoPackageScript/project_demo/AutoPackageScriptDemo/ExportOptions.plist
xcodebuild  -exportArchive \
            -archivePath ${export_archive_path} \
            -exportPath ${export_ipa_path} \
            -exportOptionsPlist ${export_options_plist_path} \
            -allowProvisioningUpdates

其中姑荷,ExportOptions.plist配置文件我只設(shè)置了method 盒延。
ExportOptions.plist配置文件

命令執(zhí)行成功之后會生成我們需要的ipa文件和其他的幾個東西。
exportArchive成功之后生成的文件

其中鼠冕,
ipa文件就是我們需要的安裝包添寺。
DistributionSummary.plist文件是一些詳細(xì)的簽名信息。
ExportOptions.plist文件其實就是我們在exportArchive命令時要用的懈费,但在exportArchive之后會自動生成一個完整的文件计露。如果不知道該怎么寫這個配置文件的話,可以直接參考我demo中生成的這個plist文件憎乙。
Packaging.log這個文件就是打包的時候產(chǎn)生的log了票罐,可以查看日志記錄。

參考

腳本和demo都放到github上面去了泞边。https://github.com/shixueqian/AutoPackageScript
本腳本主要參考了自動打包ipa文件并上傳fir.im托管平臺的shell腳本

謙言忘語

嗯该押,一個很簡單的小東西,大家有時候可以嘗試使用一些小腳本完成一些簡單重復(fù)的工作阵谚,說不定會讓生活變得更好蚕礼。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市梢什,隨后出現(xiàn)的幾起案子奠蹬,更是在濱河造成了極大的恐慌,老刑警劉巖嗡午,帶你破解...
    沈念sama閱讀 207,113評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件罩润,死亡現(xiàn)場離奇詭異,居然都是意外死亡翼馆,警方通過查閱死者的電腦和手機(jī)割以,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評論 2 381
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來应媚,“玉大人严沥,你說我怎么就攤上這事≈薪” “怎么了消玄?”我有些...
    開封第一講書人閱讀 153,340評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長丢胚。 經(jīng)常有香客問我翩瓜,道長,這世上最難降的妖魔是什么携龟? 我笑而不...
    開封第一講書人閱讀 55,449評論 1 279
  • 正文 為了忘掉前任兔跌,我火速辦了婚禮,結(jié)果婚禮上峡蟋,老公的妹妹穿的比我還像新娘坟桅。我一直安慰自己华望,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 64,445評論 5 374
  • 文/花漫 我一把揭開白布仅乓。 她就那樣靜靜地躺著赖舟,像睡著了一般。 火紅的嫁衣襯著肌膚如雪夸楣。 梳的紋絲不亂的頭發(fā)上宾抓,一...
    開封第一講書人閱讀 49,166評論 1 284
  • 那天,我揣著相機(jī)與錄音豫喧,去河邊找鬼石洗。 笑死,一個胖子當(dāng)著我的面吹牛嘿棘,可吹牛的內(nèi)容都是我干的劲腿。 我是一名探鬼主播,決...
    沈念sama閱讀 38,442評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼鸟妙,長吁一口氣:“原來是場噩夢啊……” “哼焦人!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起重父,我...
    開封第一講書人閱讀 37,105評論 0 261
  • 序言:老撾萬榮一對情侶失蹤花椭,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后房午,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體矿辽,經(jīng)...
    沈念sama閱讀 43,601評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,066評論 2 325
  • 正文 我和宋清朗相戀三年郭厌,在試婚紗的時候發(fā)現(xiàn)自己被綠了袋倔。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,161評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡折柠,死狀恐怖宾娜,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情扇售,我是刑警寧澤前塔,帶...
    沈念sama閱讀 33,792評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站承冰,受9級特大地震影響华弓,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜困乒,卻給世界環(huán)境...
    茶點故事閱讀 39,351評論 3 307
  • 文/蒙蒙 一寂屏、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦凑保、人聲如沸冈爹。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至恳谎,卻和暖如春芝此,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背因痛。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評論 1 261
  • 我被黑心中介騙來泰國打工婚苹, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人鸵膏。 一個月前我還...
    沈念sama閱讀 45,618評論 2 355
  • 正文 我出身青樓膊升,卻偏偏與公主長得像,于是被迫代替她去往敵國和親谭企。 傳聞我的和親對象是個殘疾皇子廓译,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,916評論 2 344