前言
由于項目引入了敏捷開發(fā),,需要每天build出一個ipa供QA測試。此前是使用Xcode先achive出一個文件,,再在 organizer->achives里發(fā)布ipa,,一直感覺也沒啥不方便的。直到某天,,裝了InstaSign,,突然發(fā)現(xiàn)無法用之前的方法 codesign自己的ipa(真是自作孽啊T ^ T..),網(wǎng)上有人說是修改了系統(tǒng)自帶的codesign和codesign_allocate,,重裝xcode也沒用,。不過還好能build出自己項目 的app,利用iTune,再創(chuàng)建一個ipa文件,。但是這種不得已的辦法,,對于需要每天都打ipa包的俺來說,實在是太繁瑣了,。于是就有了利用shell 腳本來創(chuàng)建ipa的想法,,也就有了此文。
正文
放狗搜了一下,,發(fā)現(xiàn)唐巧的那篇《給iOS工程增加Daily Build》比較完整的闡述了daily build的整個過程,,這里也就不贅述了。關(guān)于我關(guān)心的部分,,基本思想也很簡單:
- 利用xcodebuild,,build出程序文件<PRODUCT_NAME>.app。
- 再將程序文件<PRODUCT_NAME>.app里的所有文件,,放入Payload文件夾下,,利用zip將其打包出一個ipa文件。
失敗的思路
既然是利用shell,,剛開始,,我很自然的想到能否在xcode的build phase里添加run script,希望能在build出app后直接利用zip打包,。但是經(jīng)過測試,,發(fā)現(xiàn)腳本是在“ProcessProductPackaging(添加embedded.mobileprovision)”和“CodeSign”之前就開始運行了,顯然這個時侯zip的ipa不是有效的ipa,。
那么,,能不能直接shell也自己實現(xiàn)“ProcessProductPackaging(添加embedded.mobileprovision)”和“CodeSign”呢?
codesign還好說,,但是前者,,我實在是搞不透它用了啥內(nèi)建的工具。
無奈,,此方案作罷,。
真·解決方法
1)簡單的方法
先利用xcodebuild進行build,因為生成的目錄結(jié)構(gòu)是可知的,,所以在腳本中給變量設(shè)置好相關(guān)路徑,,參考前面介紹的那篇文章,定位到相關(guān)文件,,從而zip出ipa也是理所當(dāng)然的,。
2)蛋疼的方法
其實build時,已經(jīng)有變量可以告訴我們需要的路徑,,參考《xcode build setting reference》,,只不過這些build setting的作用范圍僅限于build階段,,也就是xcodebuild進程的執(zhí)行期間。
不過還好xcodebuild有個-showBuildSettings的參數(shù),,可以輸出相應(yīng)configuration的build setting,,那么問題的關(guān)鍵就在于如何獲取build setting并讓其作用于我用于打包的shell腳本。
注: 因為-showBuildSettings中的build dir是xcode為project生成的唯一的一個目錄,其位于~/Library/Developer/Xcode/DerivedData 下,而用腳本啟動的xcodebuild的build dir是位于腳本所在的當(dāng)前目錄,,所以還需要做一些替換,,不能獲取了直接用。
我寫的shell腳本如下:
1 # Created by chenche on 13-1-21.
2
3 #!/bin/bash
4
5 cnt=1
6 if [ $# -ne $cnt ]; then
7 echo "error param num, only allow 1 params(case sensitive)!"
8 echo "example:"
9 echo "package <configration> "
10 exit -1
11 fi
12
13 buildSettings=""
14
15 xcodebuild -configuration $1 -target <ProductName> -showBuildSettings | grep --color=never -E '=' | awk -F"=" -v currentPath=$PWD '{
16 gsub(/[[:blank:]]*/,"",$1); #去除$1中的所有blank
17 sub(/^[[:blank:]|"]*/,"",$2); #去除頭的blank,以及頭的雙引號
18 sub(/[[:blank:]|"]*$/,"", $2); #去除尾的blank,以及尾的雙引號
19
20 #print "export "$1"=\134\""$2"\134\"";
21 #print $1"=\134\""$2"\134\"";
22 if (tmp == "" && $1=="BUILD_DIR"){
23 tmp=$2;
24 sub(/\/Products$/, "/", tmp);
25 pattern=tmp"[Products|Intermediates]*";
26 #print pattern;
27 #print tmp;
28 }
29 else if (tmp !="") {
30 #如果是給gsub傳pattern參數(shù),,pattern參數(shù)的值無需在兩端加"/"
31 #pattern1 = "/Build/[Products|Intermediates]*";
32 #pattern1 = "/Build\\\//";
33 #print pattern1;
34 r = match($2, tmp);
35 if (tmp != "" && r) {
36 #print tmp" $2="$2;
37 gsub(pattern, currentPath"/build", $2);
38 #gsub(/Build\/[Products|Intermediates]*/, "00000000", $2);
39 #print $2;
40 }
41 }
42
43 print $1"="$2; #key=value
44 }' >buildTmp
45
46 while read buf
47 do
48 #echo $c
49 arr[$c]=$buf
50 let "c = $c + 1"
51 done <buildTmp
52
53 rm -rf buildTmp
54
55 #只有awk支持關(guān)聯(lián)數(shù)組,,shell本身的數(shù)組不支持,僅支持?jǐn)?shù)字的下標(biāo)
56 #echo "array len:" $c
57
58 for((i=0;i<$c;i++));
59 do
60 key=${arr[$i]/=*/}
61 value=${arr[$i]/*=/}
62
63 #UID is readonly
64 if [ "$key" != "UID" ]; then
65 # if [ -d "$value" ]; then
66 # echo $key,$value
67 # fi
68 export $key="$value"
69 fi
70 done
71
72 echo -e "\033[33;40;1m---------start building <ProductName>...---------\033[0m"
73 xcodebuild -configuration $1 -target <ProductName>
74 echo -e "\033[33;40;1m---------build over------------------------------\033[0m"
75
76 echo -e "\033[33;40;1m---------start packaging <ProductName>...--------\033[0m"
77
78 IPA_PATH=$SRCROOT/ipa
79 PAYLOAD_PATH=$IPA_PATH/Payload
80
81 mkdir -p $PAYLOAD_PATH
82 cp -r $TARGET_BUILD_DIR/$WRAPPER_NAME $PAYLOAD_PATH
83
84 cd $IPA_PATH
85 zip -r $PRODUCT_NAME.ipa *
86 mv $PRODUCT_NAME.ipa $SRCROOT
87 rm -rf $IPA_PATH
88
89 echo -e "\033[33;40;1m---------<ProductName>.ipa is done.-------------------\033[0m"
上述腳本的不足之處,大概在對于xcodebuild執(zhí)行失敗未作處理,,還是會生成一個無效的ipa,。雖然xcodebuild執(zhí)行的成功會輸出“** BUILD SUCCEEDED **”,但總感覺單純的基于這點的判斷有點不靠譜。故還是作罷了,,人工判斷好了,。
引申
寫腳本的過程中,我也碰到過一些問題,,匯總?cè)缦拢?/p>
-
普通數(shù)組和關(guān)聯(lián)數(shù)組
所謂普通數(shù)組,,下標(biāo)是數(shù)字;關(guān)聯(lián)數(shù)組類似字典,,下標(biāo)可以是數(shù)字或字符串,。網(wǎng)上搜了不少資料,都說shell支持關(guān)聯(lián)數(shù)組,,但是實際寫腳本的過程,,發(fā)現(xiàn)mac下的bash還是只支持索引數(shù)組,awk命令倒是支持關(guān)聯(lián)數(shù)組,。
另外,,可以man bash,發(fā)現(xiàn)相關(guān)內(nèi)容,,也證實了如上觀點:
An array is created automatically if any variable is assigned to using the syntax name[subscript]=value. The subscript is treated as an arithmetic expression that must evaluate to a number greater than or equal to zero.
-
環(huán)境變量
環(huán)境變量只能從父進程到子進程單向繼承,。也就是說,子進程的環(huán)境變量不會影響父進程的,。
基于1、2,,也就說明無法利用awk export相關(guān)build setting,,影響打包的shell腳本進程。
-
腳本和awk的信息交互
a 腳本->awk
- 利用export,,實現(xiàn)環(huán)境變量的單向繼承,。
- awk有個-v的參數(shù),,可以傳遞變量
b awk->腳本
- eval, 使用起來有點像javascript中的eval
- 導(dǎo)出信息到臨時文件,再利用臨時文件獲取相關(guān)信息
因為build setting里的值情況比較復(fù)雜,,最終我還是選擇了用臨時文件的方式獲取awk過濾出來的build setting信息,,再在shell腳本中export。最終,,這樣就可以利用build setting的相關(guān)值了,。
總結(jié)
好吧,其實我是在吐槽自己花了老長一段時間憋出shell腳本的艱辛歷程,。,。。雖然有點小題大做,,但好歹是復(fù)習(xí)鞏固了一下shell的相關(guān)知識,,也算沒白費勁~~~~
http://ddrccw./2013/01/29/daily-build-and-create-ipa-using-shell-script/
|