使用shell腳本build並創建ipa文件(轉)


前言

由於項目引入了敏捷開發,需要每天build出一個ipa供QA測試。此前是使用Xcode先achive出一個文件,再在 organizer->achives里發布ipa,一直感覺也沒啥不方便的。直到某天,裝了InstaSign,突然發現無法用之前的方法 codesign自己的ipa(真是自作孽啊T ^ T..),網上有人說是修改了系統自帶的codesign和codesign_allocate,重裝xcode也沒用。不過還好能build出自己項目 的app,利用iTune,再創建一個ipa文件。但是這種不得已的辦法,對於需要每天都打ipa包的俺來說,實在是太繁瑣了。於是就有了利用shell 腳本來創建ipa的想法,也就有了此文。

正文

放狗搜了一下,發現唐巧的那篇《給iOS工程增加Daily Build》比較完整的闡述了daily build的整個過程,這里也就不贅述了。關於我關心的部分,基本思想也很簡單:

  1. 利用xcodebuild,build出程序文件<PRODUCT_NAME>.app。
  2. 再將程序文件<PRODUCT_NAME>.app里的所有文件,放入Payload文件夾下,利用zip將其打包出一個ipa文件。

失敗的思路

既然是利用shell,剛開始,我很自然的想到能否在xcode的build phase里添加run script,希望能在build出app后直接利用zip打包。但是經過測試,發現腳本是在“ProcessProductPackaging(添加embedded.mobileprovision)”和“CodeSign”之前就開始運行了,顯然這個時侯zip的ipa不是有效的ipa。

那么,能不能直接shell也自己實現“ProcessProductPackaging(添加embedded.mobileprovision)”和“CodeSign”呢?

codesign還好說,但是前者,我實在是搞不透它用了啥內建的工具。

無奈,此方案作罷。

真·解決方法

1)簡單的方法

先利用xcodebuild進行build,因為生成的目錄結構是可知的,所以在腳本中給變量設置好相關路徑,參考前面介紹的那篇文章,定位到相關文件,從而zip出ipa也是理所當然的。

2)蛋疼的方法

其實build時,已經有變量可以告訴我們需要的路徑,參考《xcode build setting reference》,只不過這些build setting的作用范圍僅限於build階段,也就是xcodebuild進程的執行期間。

不過還好xcodebuild有個-showBuildSettings的參數,可以輸出相應configuration的build setting,那么問題的關鍵就在於如何獲取build setting並讓其作用於我用於打包的shell腳本。

注: 因為-showBuildSettings中的build dir是xcode為project生成的唯一的一個目錄,其位於~/Library/Developer/Xcode/DerivedData下,而用腳本啟動的xcodebuild的build dir是位於腳本所在的當前目錄,所以還需要做一些替換,不能獲取了直接用。

我寫的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參數,pattern參數的值無需在兩端加"/"
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支持關聯數組,shell本身的數組不支持,僅支持數字的下標
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執行失敗未作處理,還是會生成一個無效的ipa。雖然xcodebuild執行的成功會輸出“** BUILD SUCCEEDED **”,但總感覺單純的基於這點的判斷有點不靠譜。故還是作罷了,人工判斷好了。

引申

寫腳本的過程中,我也碰到過一些問題,匯總如下:

  1. 普通數組和關聯數組

    所謂普通數組,下標是數字;關聯數組類似字典,下標可以是數字或字符串。網上搜了不少資料,都說shell支持關聯數組,但是實際寫腳本的過程,發現mac下的bash還是只支持索引數組,awk命令倒是支持關聯數組。

    另外,可以man bash,發現相關內容,也證實了如上觀點:

    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.

  2. 環境變量

    環境變量只能從父進程到子進程單向繼承。也就是說,子進程的環境變量不會影響父進程的。

    基於1、2,也就說明無法利用awk export相關build setting,影響打包的shell腳本進程。

  3. 腳本和awk的信息交互

    a 腳本->awk

    • 利用export,實現環境變量的單向繼承。
    • awk有個-v的參數,可以傳遞變量

    b awk->腳本

    • eval, 使用起來有點像javascript中的eval
    • 導出信息到臨時文件,再利用臨時文件獲取相關信息

因為build setting里的值情況比較復雜,最終我還是選擇了用臨時文件的方式獲取awk過濾出來的build setting信息,再在shell腳本中export。最終,這樣就可以利用build setting的相關值了。

總結

好吧,其實我是在吐槽自己花了老長一段時間憋出shell腳本的艱辛歷程。。。雖然有點小題大做,但好歹是復習鞏固了一下shell的相關知識,也算沒白費勁~~~~

http://ddrccw.github.io/2013/01/29/daily-build-and-create-ipa-using-shell-script/


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM