package_utils.py 43 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275
  1. # 安卓游戏打包脚本
  2. # 1.用apktool解包
  3. # 2.复制res资源、assets资源、jniLib资源
  4. # 3.修改包名、app名、渠道文件
  5. # 4.添加app icon、合并AndroidManifest文件
  6. # 5.添加app启动图,修改AndroidManifest文件
  7. # 6.生成新的R文件
  8. # 7.将新的R文件编译,生成dex,再用baksmali生成smali,替换旧的R文件
  9. # 8.将jar资源打包成dex,再用baksmali生成smali,拷贝到smali目录下
  10. # 9.统计方法量,并分割dex(将smali文件拷贝到目录smali_classes2)
  11. # 10.用apktool回包
  12. # 11.重新签名
  13. import file_utils
  14. import xml_utils
  15. import smali_utils
  16. import config_utils
  17. import game_utils
  18. import os
  19. import os.path
  20. import json
  21. import sys
  22. import importlib
  23. import uuid
  24. import zipfile
  25. ignoreLauncher = ['jm_ysdk', 'jm_yijie', 'jm_quick', 'jm_beiyu', 'jm_xq_jrtt']
  26. adaptApp = ['yjzx']
  27. def pack(game, sdk, config):
  28. config['cache'] = uuid.uuid1()
  29. subChannel = config['subChannel']
  30. # 解包
  31. ret = decomplie(game, sdk, subChannel, config)
  32. if ret:
  33. return ret
  34. # 删除旧代码
  35. ret = removeOldCode(game, sdk, subChannel, config)
  36. if ret:
  37. return ret
  38. # 删除一些不支持的属性
  39. ret = removeNoSupportAttr(game, sdk, subChannel, config)
  40. if ret:
  41. return ret
  42. # 删除一些不支持的配置
  43. ret = fixUnSupportConfig(game, sdk, subChannel, config)
  44. if ret:
  45. return ret
  46. # 合并Drawable-v4目录
  47. ret = mergeDrawableRes(game, sdk, subChannel, config)
  48. if ret:
  49. return ret
  50. # 移除相同的资源
  51. ret = removeSameRes(game, sdk, subChannel, config)
  52. if ret:
  53. return ret
  54. # 复制res资源
  55. ret = copyRes(game, sdk, subChannel, config)
  56. if ret:
  57. return ret
  58. # 合并主文件
  59. ret = mergeManifestRes(game, sdk, subChannel, config)
  60. if ret:
  61. return ret
  62. # 替换占位符
  63. ret = changePlaceholders(game, sdk, subChannel, config)
  64. if ret:
  65. return ret
  66. # 添加meta-data
  67. ret = addMetaData(game, sdk, subChannel, config)
  68. if ret:
  69. return ret
  70. # 增加配置文件
  71. ret = addConfig(game, sdk, subChannel, config)
  72. if ret:
  73. return ret
  74. # 复制app res资源
  75. ret = copyAppRes(game, sdk, subChannel, config)
  76. if ret:
  77. return ret
  78. # 更改包名
  79. if 'packageName' in config and config['packageName'] != '':
  80. ret = changePackageName(game, sdk, subChannel, config)
  81. if ret:
  82. return ret
  83. # 更改app名
  84. if 'name' in config and config['name'] != '':
  85. ret = changeAppName(game, sdk, subChannel, config)
  86. if ret:
  87. return ret
  88. # 更改app icon
  89. if config['changeIcon']:
  90. ret = changeAppIcon(game, sdk, subChannel, config)
  91. if ret:
  92. return ret
  93. # 添加启动图操作
  94. if config['addLauncher']:
  95. ret = addLauncher(game, sdk, subChannel, config)
  96. if ret:
  97. return ret
  98. # 添加多图标
  99. #ret = addMoreIcon(game, sdk, subChannel, config)
  100. # 复制icon图标到res
  101. ret = copyIcon(game, sdk, subChannel, config)
  102. #if ret:
  103. # return ret
  104. # 打包lib依赖
  105. ret = packJar(game, sdk, subChannel, config)
  106. if ret:
  107. return ret
  108. # sdk脚本处理
  109. ret = doSDKPostScript(game, sdk, config)
  110. if ret:
  111. return ret
  112. # 乐变sdk的特殊处理
  113. ret = game_utils.sdkLebianChange(game, sdk, config)
  114. if ret:
  115. return ret
  116. # 游戏脚本处理
  117. ret = doGamePostScript(game, sdk, config)
  118. if ret:
  119. return ret
  120. # log sdk
  121. if 'logSdk' in config:
  122. for log in config['logSdk']:
  123. ret = addLogSdk(game, sdk, subChannel, config, log)
  124. if ret:
  125. return ret
  126. # 生成R文件
  127. '''ret = generateNewRFile(game, sdk, subChannel, config)
  128. if ret:
  129. return ret'''
  130. # 添加MultiDex支持
  131. if config['splitDex']:
  132. ret = splitDex(game, sdk, subChannel, config)
  133. if ret:
  134. return ret
  135. # 更改版本号
  136. ret = changeVersion(game, sdk, subChannel, config)
  137. if ret:
  138. return ret
  139. # 回编译
  140. ret = recomplie(game, sdk, subChannel, config)
  141. if ret:
  142. return ret
  143. # 对齐apk
  144. ret = alignApk(game, sdk, subChannel, config)
  145. if ret:
  146. return ret
  147. # 签名
  148. ret = apksignerApk(game, sdk, subChannel, config)
  149. if ret:
  150. return ret
  151. # 添加渠道信息
  152. ret = addChannel(game, sdk, subChannel, config)
  153. if ret:
  154. return ret
  155. # 清理产生的中间文件
  156. if config['clearCache']:
  157. clearTemp(game, sdk, subChannel, config)
  158. return 0
  159. def decomplie(game, sdk, subChannel, config):
  160. '''
  161. 解包
  162. '''
  163. apktoolPath = file_utils.getApkToolPath()
  164. gamePath = file_utils.getFullGameApk(game)
  165. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  166. if os.path.exists(decompliePath):
  167. print('delete decomplie folder...')
  168. file_utils.deleteFolder(decompliePath)
  169. print('decomplie apk...')
  170. return file_utils.execJarCmd(apktoolPath, 'd -f "%s" -o "%s"' % (gamePath, decompliePath))
  171. def removeOldCode(game, sdk, subChannel, config):
  172. '''
  173. 删除旧代码
  174. '''
  175. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  176. codePath = os.path.join(decompliePath, 'smali', 'com', 'jmhy', 'sdk')
  177. #file_utils.deleteFolder(codePath)
  178. allFiles = []
  179. allFiles = file_utils.list_files(codePath, allFiles)
  180. for f in allFiles:
  181. fpath, fname = os.path.split(f) #分离文件名和路径
  182. if fname == 'R.smali' or fname.startswith('R$'):
  183. continue
  184. os.remove(f)
  185. print('remove %s' % f)
  186. return 0
  187. def copyRes(game, sdk, subChannel, config):
  188. '''
  189. 复制res资源
  190. '''
  191. # 拷贝sdk资源
  192. print('copy res...')
  193. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  194. sdkPath = file_utils.getFullSDKPath(sdk)
  195. resPath = os.path.join(sdkPath, 'res')
  196. decomplieResPath = file_utils.getFullPath(decompliePath, 'res')
  197. for d in os.listdir(resPath):
  198. copyResWithType(resPath, decomplieResPath, d)
  199. # 拷贝assets
  200. print('copy assets...')
  201. assetsPath = file_utils.getFullPath(sdkPath, 'assets')
  202. decomplieAssetsPath = file_utils.getFullPath(decompliePath, 'assets')
  203. if os.path.exists(assetsPath):
  204. ret = file_utils.copyFileAllDir(assetsPath, decomplieAssetsPath)
  205. if ret:
  206. return ret
  207. # 拷贝jniLib
  208. print('copy jniLibs...')
  209. jniPath = file_utils.getFullPath(sdkPath, 'jniLibs')
  210. decomplieJniPath = file_utils.getFullPath(decompliePath, 'lib')
  211. abiFilters = []
  212. if os.path.exists(decomplieJniPath):
  213. for abi in os.listdir(decomplieJniPath):
  214. if abi == 'armeabi-v7a' or abi == 'armeabi':
  215. abiFilters.append(abi)
  216. else:
  217. abiFilters = ['armeabi-v7a']
  218. if os.path.exists(jniPath):
  219. ret = file_utils.copyFileAllDir(jniPath, decomplieJniPath, False, abiFilters)
  220. if ret:
  221. return ret
  222. return 0
  223. def mergeDrawableRes(game, sdk, subChannel, config):
  224. '''
  225. 合并Drawable-v4目录
  226. '''
  227. print('merge drawable path...')
  228. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  229. resPath = os.path.join(decompliePath, 'res')
  230. for path in os.listdir(resPath):
  231. print('res path %s' % path)
  232. if (path.startswith('drawable') or path.startswith('mipmap')) and path.endswith('-v4'):
  233. print('merge path %s' % path)
  234. v4DrawablePath = os.path.join(resPath, path)
  235. drawablePath = os.path.join(resPath, path[:-3])
  236. if os.path.exists(drawablePath):
  237. ret = file_utils.copyFileAllDir(v4DrawablePath, drawablePath, True)
  238. if ret:
  239. return ret
  240. else:
  241. os.rename(v4DrawablePath, drawablePath)
  242. return 0
  243. def removeSameRes(game, sdk, subChannel, config):
  244. '''
  245. 移除相同的资源
  246. '''
  247. # 读取sdk的资源
  248. print('remove same res...')
  249. sdkPath = file_utils.getFullSDKPath(sdk)
  250. sdkResPath = os.path.join(sdkPath, 'res')
  251. resList = []
  252. for path in os.listdir(sdkResPath):
  253. if not path.startswith('values'):
  254. continue
  255. absPath = os.path.join(sdkResPath, path)
  256. for resFile in os.listdir(absPath):
  257. '''if not resFile.startswith('jm_'):
  258. continue'''
  259. resList = xml_utils.readAllRes(os.path.join(absPath, resFile), resList)
  260. if len(resList) == 0:
  261. print('no same res found')
  262. return 0
  263. # 移除相同的资源
  264. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  265. resPath = os.path.join(decompliePath, 'res')
  266. for path in os.listdir(resPath):
  267. if not path.startswith('values'):
  268. continue
  269. absPath = os.path.join(resPath, path)
  270. for resFile in os.listdir(absPath):
  271. xml_utils.removeSameRes(os.path.join(absPath, resFile), resList)
  272. return 0
  273. def mergeManifestRes(game, sdk, subChannel, config):
  274. '''
  275. 合并主文件
  276. '''
  277. print('merge AndroidManifest...')
  278. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  279. manifest = os.path.join(decompliePath, 'AndroidManifest.xml')
  280. sdkPath = file_utils.getFullSDKPath(sdk)
  281. libManifest = file_utils.getFullPath(sdkPath, 'manifest.xml')
  282. return xml_utils.mergeManifestRes(manifest, libManifest)
  283. def copyAppRes(game, sdk, subChannel, config):
  284. '''
  285. 拷贝app的资源,比如app icon、启动图等
  286. '''
  287. channelPath = file_utils.getSubChannelPath(game, sdk, subChannel)
  288. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  289. # assets
  290. print('copy assets...')
  291. #适配一剑斩仙
  292. if config['app'] in adaptApp:
  293. print('适配一剑斩仙...')
  294. decomplieAssetsPath = file_utils.getFullPath(decompliePath, 'assets')
  295. skinZipPath = os.path.join(decomplieAssetsPath, 'skin.zip')
  296. skinPath = os.path.join(decomplieAssetsPath, 'skin')
  297. if os.path.exists(skinZipPath):
  298. with zipfile.ZipFile(skinZipPath) as zf:
  299. zf.extractall(decomplieAssetsPath)
  300. print('create unzip skin' )
  301. assetsPath = file_utils.getFullPath(channelPath, 'assets')
  302. decomplieAssetsPath = file_utils.getFullPath(decompliePath, 'assets')
  303. if os.path.exists(assetsPath):
  304. ret = file_utils.copyFileAllDir(assetsPath, decomplieAssetsPath)
  305. with zipfile.ZipFile(skinZipPath, 'w') as z:
  306. for root, dirs, files in os.walk(skinPath):
  307. for single_file in files:
  308. filepath = os.path.join(root, single_file)
  309. print ('create zip ---> ' + filepath)
  310. temPath = 'skin/' + single_file
  311. z.write(filepath, temPath)
  312. z.close()
  313. else:
  314. print('normal copy assets...')
  315. assetsPath = file_utils.getFullPath(channelPath, 'assets')
  316. decomplieAssetsPath = file_utils.getFullPath(decompliePath, 'assets')
  317. if os.path.exists(assetsPath):
  318. ret = file_utils.copyFileAllDir(assetsPath, decomplieAssetsPath)
  319. if ret:
  320. return ret
  321. else:
  322. print('normal copy assets...')
  323. assetsPath = file_utils.getFullPath(channelPath, 'assets')
  324. decomplieAssetsPath = file_utils.getFullPath(decompliePath, 'assets')
  325. if os.path.exists(assetsPath):
  326. ret = file_utils.copyFileAllDir(assetsPath, decomplieAssetsPath)
  327. if ret:
  328. return ret
  329. # icon
  330. print('copy icon...')
  331. ret = copyAppResWithType(decompliePath, channelPath, 'icon')
  332. if ret:
  333. return ret
  334. # 启动图
  335. print('copy splash...')
  336. ret = copyAppResWithType(decompliePath, channelPath, 'splash')
  337. if ret:
  338. return ret
  339. # 其他图片
  340. print('copy image...')
  341. ret = copyAppResWithType(decompliePath, channelPath, 'image')
  342. if ret:
  343. return ret
  344. return 0
  345. def copyAppResWithType(decompliePath, channelPath, typeName):
  346. decomplieResPath = os.path.join(decompliePath, 'res')
  347. iconPath = os.path.join(channelPath, typeName)
  348. if not os.path.exists(iconPath):
  349. print('dir "%s" not exists' % iconPath)
  350. return 0
  351. for d in os.listdir(iconPath):
  352. ret = copyResWithType(iconPath, decomplieResPath, d)
  353. if ret:
  354. return ret
  355. return 0
  356. def copyResWithType(resPath, decomplieResPath, typeName):
  357. # appt的打包目录会带-v4后缀
  358. resDir = os.path.join(resPath, typeName)
  359. target = os.path.join(decomplieResPath, typeName)
  360. targetV4 = os.path.join(decomplieResPath, typeName + '-v4')
  361. if not os.path.exists(target) and not os.path.exists(targetV4):
  362. os.makedirs(target)
  363. return file_utils.copyFileAllDir(resDir, target, False)
  364. elif not os.path.exists(target):
  365. return file_utils.copyFileAllDir(resDir, targetV4, False)
  366. else:
  367. return file_utils.copyFileAllDir(resDir, target, False)
  368. def removeNoSupportAttr(game, sdk, subChannel, config):
  369. '''
  370. 删除一些不支持的属性
  371. '''
  372. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  373. manifest = os.path.join(decompliePath, 'AndroidManifest.xml')
  374. xml_utils.removeRootAttr(manifest, 'compileSdkVersion')
  375. xml_utils.removeRootAttr(manifest, 'compileSdkVersionCodename')
  376. return 0
  377. def fixUnSupportConfig(game, sdk, subChannel, config):
  378. '''
  379. 删除一些不支持的配置
  380. '''
  381. # 检查minSdkVersion
  382. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  383. yml = os.path.join(decompliePath, 'apktool.yml')
  384. minSdkVersion = 15
  385. file_utils.changeMinSdkVersion(yml, minSdkVersion)
  386. resPath = os.path.join(decompliePath, 'res')
  387. tag = '-v'
  388. for res in os.listdir(resPath):
  389. print('res = ' + res)
  390. if res.startswith('values') and tag in res:
  391. start = res.index(tag)
  392. version = res[start+len(tag):]
  393. if not version.isdigit():
  394. continue
  395. version = int(version)
  396. print('version = %d' % version)
  397. if version < minSdkVersion:
  398. unSopportPath = os.path.join(resPath, res)
  399. print('unSopportPath = ' + unSopportPath)
  400. file_utils.deleteFolder(unSopportPath)
  401. print('deleteFolder = ' + unSopportPath)
  402. return 0
  403. def changePackageName(game, sdk, subChannel, config):
  404. '''
  405. 更改包名
  406. '''
  407. # 全局替换AndroidManifest里面的包名
  408. newPackageName = config['packageName']
  409. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  410. manifest = os.path.join(decompliePath, 'AndroidManifest.xml')
  411. packageName = xml_utils.getPackageName(manifest)
  412. xml_utils.changePackageName(manifest, newPackageName)
  413. print('change package name %s --> %s' % (packageName, newPackageName))
  414. return 0
  415. def changeAppName(game, sdk, subChannel, config):
  416. '''
  417. 更改app名
  418. '''
  419. # 生成string.xml文件
  420. name = config['name']
  421. resName = 'common_sdk_name'
  422. if 'outName' in config:
  423. resName = resName + '_' + config['outName']
  424. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  425. stringFile = os.path.join(decompliePath, 'res', 'values', 'sdk_strings_temp.xml')
  426. content = '<?xml version="1.0" encoding="utf-8"?><resources><string name="%s">%s</string></resources>' % (resName, name)
  427. file_utils.createFile(stringFile, content)
  428. # 修改主文件的app名的值
  429. manifest = os.path.join(decompliePath, 'AndroidManifest.xml')
  430. xml_utils.changeAppName(manifest, '@string/%s' % resName)
  431. print('change app name %s' % name)
  432. return 0
  433. def changeAppIcon(game, sdk, subChannel, config):
  434. '''
  435. 更改app icon
  436. '''
  437. print('change app icon...')
  438. resName = 'common_sdk_icon'
  439. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  440. manifest = os.path.join(decompliePath, 'AndroidManifest.xml')
  441. xml_utils.changeAppIcon(manifest, '@mipmap/%s' % resName)
  442. return 0
  443. def addMetaData(game, sdk, subChannel, config):
  444. '''
  445. 添加meta-data
  446. '''
  447. if 'metaData' not in config:
  448. return 0
  449. print('add meta-data...')
  450. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  451. manifest = os.path.join(decompliePath, 'AndroidManifest.xml')
  452. xml_utils.addMetaData(manifest, config['metaData'])
  453. return 0
  454. def addConfig(game, sdk, subChannel, config):
  455. '''
  456. 添加config.json
  457. '''
  458. if 'configData' not in config:
  459. return 0
  460. print('add config.json...')
  461. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  462. configJson = os.path.join(decompliePath, 'assets', 'jmhy_config.json')
  463. jsonText = json.dumps(config['configData'], ensure_ascii=False)
  464. file_utils.createFile(configJson, jsonText)
  465. return 0
  466. def changePlaceholders(game, sdk, subChannel, config):
  467. '''
  468. 处理掉占位符
  469. '''
  470. if 'placeholders' not in config:
  471. return 0
  472. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  473. manifest = os.path.join(decompliePath, 'AndroidManifest.xml')
  474. placeholders = config['placeholders']
  475. for placeholder in placeholders:
  476. oldText = '${%s}' % placeholder
  477. newText = placeholders[placeholder]
  478. print('change placeholder %s -> %s' % (oldText, newText))
  479. file_utils.replaceContent(manifest, oldText, newText)
  480. return 0
  481. def addLauncher(game, sdk, subChannel, config):
  482. '''
  483. 添加启动图
  484. '''
  485. # ysdk的特殊处理
  486. if sdk in ignoreLauncher:
  487. return 0
  488. channelPath = file_utils.getSubChannelPath(game, sdk, subChannel)
  489. splashPath = os.path.join(channelPath, 'splash')
  490. if len(os.listdir(splashPath)) == 0:
  491. print('dir splash is empty')
  492. return 0
  493. print('add launcher...')
  494. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  495. manifest = os.path.join(decompliePath, 'AndroidManifest.xml')
  496. activity = xml_utils.getLauncherActivityName(manifest)
  497. if activity == 'com.jmhy.sdk.template.LauncherActivity':
  498. print('add launcher already exist...')
  499. return 1
  500. # 添加关联资源
  501. internalPath = file_utils.getFullInternalPath()
  502. ret = copyAppResWithType(decompliePath, internalPath, 'launcher_res')
  503. if ret:
  504. return ret
  505. # 拷贝代码
  506. print('copy launcher code...')
  507. codePath = os.path.join(internalPath, 'launcher_code', 'smali')
  508. smaliPath = file_utils.getFullPath(decompliePath, 'smali')
  509. ret = file_utils.copyFileAllDir(codePath, smaliPath)
  510. if ret:
  511. return ret
  512. # 修改主文件信息
  513. print('change launcher config...')
  514. orientation = xml_utils.getScreenOrientation(manifest)
  515. activity = xml_utils.removeLauncherActivity(manifest)
  516. xml_utils.addLauncherActivity(manifest, orientation, 'com.jmhy.sdk.template.LauncherActivity')
  517. # 修改跳转的
  518. launcherActivity = os.path.join(decompliePath, 'smali', 'com', 'jmhy', 'sdk', 'template', 'LauncherActivity.smali')
  519. file_utils.replaceContent(launcherActivity, '{class}', activity)
  520. print('change launcher %s to %s' % (activity, 'com.jmhy.sdk.template.LauncherActivity'))
  521. # config['oldLauncher'] = activity
  522. if 'launcherTime' in config:
  523. timeHex = formatHex(config['launcherTime'])
  524. file_utils.replaceContent(launcherActivity, '0x0BB8', timeHex)
  525. return 0
  526. def addMoreIcon(game, sdk, subChannel, config):
  527. '''
  528. 添加多个图标
  529. '''
  530. print('add more icon support...')
  531. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  532. icon = '@mipmap/common_sdk_icon'
  533. if not config['changeIcon']:
  534. manifest = os.path.join(decompliePath, 'AndroidManifest.xml')
  535. icon = xml_utils.getApplicationAttr(manifest, 'icon')
  536. switchIcon = icon
  537. if config['switchIcon']:
  538. switchIcon = '@mipmap/common_sdk_icon2'
  539. manifest = os.path.join(decompliePath, 'AndroidManifest.xml')
  540. return xml_utils.addMoreIcon(manifest, icon, switchIcon)
  541. def copyIcon(game, sdk, subChannel, config):
  542. '''
  543. 复制icon到res ,一键登录使用
  544. '''
  545. if config['changeIcon']:
  546. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  547. decomplieResPath = file_utils.getFullPath(decompliePath, 'res')
  548. iconPath = os.path.join(decomplieResPath, 'mipmap-xhdpi', 'common_sdk_icon.png')
  549. desPath = os.path.join(decomplieResPath, 'drawable-hdpi', 'jm_cmcc_icon.png')
  550. file_utils.copyFile(iconPath, desPath)
  551. else:
  552. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  553. decomplieResPath = file_utils.getFullPath(decompliePath, 'res')
  554. desPath = os.path.join(decomplieResPath, 'drawable-hdpi', 'jm_cmcc_icon.png')
  555. manifest = os.path.join(decompliePath, 'AndroidManifest.xml')
  556. icon = xml_utils.getApplicationAttr(manifest, 'icon')
  557. tag = icon[1:].split("/")
  558. if len(tag) < 1:
  559. return
  560. hdpi = ['-xhdpi', '-xxhdpi', '-xxxhdpi']
  561. for h in hdpi:
  562. p1 = '%s%s' % (tag[0],h)
  563. png = tag[1] + ".png"
  564. iconPath = os.path.join(decomplieResPath, p1, png)
  565. if os.path.exists(iconPath):
  566. print ('iconPath = ' + iconPath)
  567. file_utils.copyFile(iconPath, desPath)
  568. break
  569. def formatHex(millisecond):
  570. '''
  571. 将毫秒转为16进制,4位格式
  572. '''
  573. timeHex = str(hex(millisecond)).upper()
  574. timeHex = timeHex[2:]
  575. formatHex = ''
  576. if len(timeHex) == 3:
  577. formatHex = '0x0' + timeHexaddLauncher
  578. elif len(timeHex) == 4:
  579. formatHex = '0x' + timeHex
  580. else:
  581. formatHex = '0x0BB8'
  582. return formatHex
  583. def doSDKPostScript(game, sdk, config):
  584. '''
  585. 执行sdk相关特殊处理脚本
  586. '''
  587. sdkPath = file_utils.getFullSDKPath(sdk)
  588. scriptPath = os.path.join(sdkPath, 'script')
  589. targetScript = os.path.join(scriptPath, 'sdk_script.py')
  590. if not os.path.exists(targetScript):
  591. print('sdk_script no exists')
  592. return 0
  593. print('doSDKPostScript...')
  594. sys.path.append(scriptPath)
  595. module = importlib.import_module('sdk_script')
  596. ret = module.execute(game, sdk, config)
  597. sys.path.remove(scriptPath)
  598. return ret
  599. def doGamePostScript(game, sdk, config):
  600. '''
  601. 执行游戏相关特殊处理脚本
  602. '''
  603. channelPath = file_utils.getFullGamePath(game)
  604. scriptPath = os.path.join(channelPath, 'script')
  605. targetScript = os.path.join(scriptPath, 'game_script.py')
  606. if not os.path.exists(targetScript):
  607. print('game_script no exists')
  608. return 0
  609. print('doGamePostScript...')
  610. sys.path.append(scriptPath)
  611. module = importlib.import_module('game_script')
  612. ret = module.execute(game, sdk, config)
  613. sys.path.remove(scriptPath)
  614. return ret
  615. def addLogSdk(game, sdk, subChannel, config, logSdk):
  616. # 拷贝jniLibs
  617. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  618. sdkPath = file_utils.getFullLogSDKPath(logSdk)
  619. print('copy log jniLibs...')
  620. jniPath = file_utils.getFullPath(sdkPath, 'jniLibs')
  621. decomplieJniPath = file_utils.getFullPath(decompliePath, 'lib')
  622. abiFilters = []
  623. if os.path.exists(decomplieJniPath):
  624. for abi in os.listdir(decomplieJniPath):
  625. if abi == 'armeabi-v7a' or abi == 'armeabi':
  626. abiFilters.append(abi)
  627. else:
  628. abiFilters = ['armeabi-v7a']
  629. if os.path.exists(jniPath):
  630. ret = file_utils.copyFileAllDir(jniPath, decomplieJniPath, False, abiFilters)
  631. if ret:
  632. return ret
  633. print('merge log AndroidManifest...')
  634. libManifest = file_utils.getFullPath(sdkPath, 'manifest.xml')
  635. if os.path.exists(libManifest):
  636. manifest = os.path.join(decompliePath, 'AndroidManifest.xml')
  637. ret = xml_utils.mergeManifestRes(manifest, libManifest)
  638. if ret:
  639. return ret
  640. return packLogJar(game, sdk, subChannel, config, logSdk)
  641. def generateNewRFile(game, sdk, subChannel, config):
  642. '''
  643. 生成新的R文件
  644. '''
  645. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  646. androidPlatforms = file_utils.getAndroidCompileToolPath()
  647. manifest = os.path.join(decompliePath, 'AndroidManifest.xml')
  648. decomplieResPath = file_utils.getFullPath(decompliePath, 'res')
  649. compliePath = file_utils.getFullPath(decompliePath, 'gen')
  650. if not os.path.exists(compliePath):
  651. os.makedirs(compliePath)
  652. # 生成R文件
  653. print('create R.java ...')
  654. if config['aapt2disable']:
  655. aapt = file_utils.getAAPTPath()
  656. ret = file_utils.getExecPermission(aapt)
  657. if ret:
  658. return ret
  659. createRCmd = '"%s" p -f -m -J "%s" -S "%s" -I "%s" -M "%s"' % (aapt, compliePath, decomplieResPath, androidPlatforms, manifest)
  660. ret = file_utils.execFormatCmd(createRCmd)
  661. if ret:
  662. return ret
  663. else:
  664. # compile
  665. aapt = file_utils.getAAPT2Path()
  666. ret = file_utils.getExecPermission(aapt)
  667. if ret:
  668. return ret
  669. print('compiled res ...')
  670. complieResPath = os.path.join(compliePath, 'compiled')
  671. complieResCmd = '"%s" compile --dir "%s" -o "%s"' % (aapt, decomplieResPath, complieResPath)
  672. file_utils.execFormatCmd(complieResCmd)
  673. # unzip
  674. print('unzip compiled res ...')
  675. unzipResPath = os.path.join(compliePath, 'aapt2_res')
  676. with zipfile.ZipFile(complieResPath) as zf:
  677. zf.extractall(unzipResPath)
  678. print('create unzip %s' % unzipResPath)
  679. # link
  680. print('link res ...')
  681. outApk = os.path.join(compliePath, 'res.apk')
  682. linkResCmd = '"%s" link -o "%s" -I "%s" --manifest "%s" --java "%s" --auto-add-overlay' % (aapt, outApk, androidPlatforms, manifest, compliePath)
  683. for filename in os.listdir(unzipResPath):
  684. linkResCmd += ' %s' % filename
  685. print('link cmd len is %s' % len(linkResCmd))
  686. ret = file_utils.execFormatCmd(linkResCmd, unzipResPath)
  687. if ret:
  688. return ret
  689. # 编译R文件
  690. print('complie R.java ...')
  691. packageName = xml_utils.getPackageName(manifest)
  692. packagePath = file_utils.getPackagePath(compliePath, packageName)
  693. RSourceFile = os.path.join(packagePath, 'R.java')
  694. complieRCmd = 'javac -source 1.8 -target 1.8 -encoding UTF-8 "%s"' % RSourceFile
  695. ret = file_utils.execFormatCmd(complieRCmd)
  696. if ret:
  697. return ret
  698. # 生成dex
  699. print('dex R.class ...')
  700. outDex = os.path.join(compliePath, 'classes.dex')
  701. if config['aapt2disable']:
  702. dx = file_utils.getDxPath()
  703. dexCmd = '--dex --no-warning --output="%s" "%s"' % (outDex, compliePath)
  704. else:
  705. dx = file_utils.getD8Path()
  706. clazz = os.path.join(packagePath, '*.class')
  707. dexCmd = '--lib "%s" --output "%s" %s' % (androidPlatforms, compliePath, clazz)
  708. ret = file_utils.execJarCmd(dx, dexCmd)
  709. if ret:
  710. return ret
  711. # 反向dex生成smali
  712. # 存放在out目录
  713. print('baksmali classes.dex ...')
  714. baksmaliPath = file_utils.getBaksmaliPath()
  715. outPath = file_utils.getFullPath(decompliePath, 'out')
  716. ret = file_utils.execJarCmd(baksmaliPath, 'd "%s" -o "%s"' % (outDex, outPath))
  717. if ret:
  718. return ret
  719. # 将生成的文件拷贝到目标目录
  720. print('copy R.smali ...')
  721. smaliPath = file_utils.getFullPath(decompliePath, 'smali')
  722. file_utils.copyFileAllDir(outPath, smaliPath)
  723. return 0
  724. def packJar(game, sdk, subChannel, config):
  725. '''
  726. 打包所有的jar
  727. '''
  728. splitDex = config['splitDex']
  729. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  730. outPath = file_utils.getFullPath(decompliePath, 'gen')
  731. if not os.path.exists(outPath):
  732. os.makedirs(outPath)
  733. if config['aapt2disable']:
  734. dx = file_utils.getDxPath()
  735. dexCmd = '--dex --multi-dex --no-warning --output="%s"' % outPath
  736. else:
  737. dx = file_utils.getD8Path()
  738. androidPlatforms = file_utils.getAndroidCompileToolPath()
  739. dexCmd = '--lib "%s" --output "%s"' % (androidPlatforms, outPath)
  740. # 找到所有lib依赖
  741. sdkPath = file_utils.getFullSDKPath(sdk)
  742. libs = os.path.join(sdkPath, 'libs')
  743. libConfig = os.path.join(libs, 'config.json')
  744. # 存在配置文件
  745. if os.path.exists(libConfig):
  746. jsonText = file_utils.readFile(libConfig)
  747. libConf = json.loads(jsonText)
  748. if 'libConfig' in config and config['libConfig'] in libConf:
  749. conf = config['libConfig']
  750. libList = libConf[conf]
  751. for jar in libList:
  752. dexCmd += ' ' + os.path.join(libs, jar)
  753. elif 'default' in libConf:
  754. libList = libConf['default']
  755. for jar in libList:
  756. dexCmd += ' ' + os.path.join(libs, jar)
  757. else:
  758. for jar in os.listdir(libs):
  759. if not jar.endswith('.jar'):
  760. continue
  761. dexCmd += ' ' + os.path.join(libs, jar)
  762. else:
  763. for jar in os.listdir(libs):
  764. if not jar.endswith('.jar'):
  765. continue
  766. dexCmd += ' ' + os.path.join(libs, jar)
  767. # multidex.jar
  768. if splitDex:
  769. dexCmd += ' ' + file_utils.getMultiDexPath()
  770. # sdk实现类
  771. print('packageing all jar ...')
  772. dexCmd += ' ' + os.path.join(sdkPath, '%s.jar' % sdk)
  773. ret = file_utils.execJarCmd(dx, dexCmd)
  774. if ret:
  775. return ret
  776. # 反向dex生成smali
  777. # 存放在out目录
  778. print('baksmali classes.dex ...')
  779. outDex = os.path.join(outPath, 'classes.dex')
  780. baksmaliPath = file_utils.getBaksmaliPath()
  781. outPath = file_utils.getFullPath(decompliePath, 'out')
  782. ret = file_utils.execJarCmd(baksmaliPath, 'd "%s" -o "%s"' % (outDex, outPath))
  783. if ret:
  784. return ret
  785. # 将生成的文件拷贝到目标目录
  786. print('copy all smali ...')
  787. smaliPath = file_utils.getFullPath(decompliePath, 'smali')
  788. ret = file_utils.copyFileAllDir(outPath, smaliPath, True)
  789. if ret:
  790. return ret
  791. return 0
  792. def packLogJar(game, sdk, subChannel, config, logSdk):
  793. '''
  794. 打包所有的jar
  795. '''
  796. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  797. outPath = file_utils.getFullPath(decompliePath, 'gen')
  798. if not os.path.exists(outPath):
  799. os.makedirs(outPath)
  800. if config['aapt2disable']:
  801. dx = file_utils.getDxPath()
  802. dexCmd = '--dex --multi-dex --no-warning --output="%s"' % outPath
  803. else:
  804. dx = file_utils.getD8Path()
  805. androidPlatforms = file_utils.getAndroidCompileToolPath()
  806. dexCmd = '--lib "%s" --output "%s"' % (androidPlatforms, outPath)
  807. # 找到所有lib依赖
  808. sdkPath = file_utils.getFullLogSDKPath(logSdk)
  809. libs = os.path.join(sdkPath, 'libs')
  810. libConfig = os.path.join(libs, 'config.json')
  811. # 存在配置文件
  812. if os.path.exists(libConfig):
  813. jsonText = file_utils.readFile(libConfig)
  814. libList = json.loads(jsonText)
  815. for jar in libList:
  816. if not jar.endswith('.jar'):
  817. continue
  818. dexCmd += ' ' + os.path.join(libs, jar)
  819. else:
  820. for jar in os.listdir(libs):
  821. if not jar.endswith('.jar'):
  822. continue
  823. dexCmd += ' ' + os.path.join(libs, jar)
  824. # sdk实现类
  825. print('packageing all log jar ...')
  826. dexCmd += ' ' + os.path.join(sdkPath, '%s.jar' % logSdk)
  827. ret = file_utils.execJarCmd(dx, dexCmd)
  828. if ret:
  829. return ret
  830. # 反向dex生成smali
  831. # 存放在out目录
  832. print('baksmali classes.dex ...')
  833. outDex = os.path.join(outPath, 'classes.dex')
  834. baksmaliPath = file_utils.getBaksmaliPath()
  835. outPath = file_utils.getFullPath(decompliePath, 'out')
  836. ret = file_utils.execJarCmd(baksmaliPath, 'd "%s" -o "%s"' % (outDex, outPath))
  837. if ret:
  838. return ret
  839. # 将生成的文件拷贝到目标目录
  840. print('copy all log smali ...')
  841. smaliPath = file_utils.getFullPath(decompliePath, 'smali')
  842. ret = file_utils.copyFileAllDir(outPath, smaliPath, True)
  843. if ret:
  844. return ret
  845. return 0
  846. def splitDex(game, sdk, subChannel, config):
  847. '''
  848. 分割dex
  849. '''
  850. # 判断是否已经存在application
  851. # 存在,则往原application添加内容
  852. # 不存在,则拷贝一个默认的android.support.multidex.MultiDexApplication
  853. print('add MultiDex support...')
  854. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  855. manifest = os.path.join(decompliePath, 'AndroidManifest.xml')
  856. application = xml_utils.getApplicationAttr(manifest, 'name')
  857. if application is None:
  858. ret = xml_utils.changeApplicationAttr(manifest, 'name', 'android.support.multidex.MultiDexApplication')
  859. if ret:
  860. return ret
  861. else:
  862. smaliPath = os.path.join(decompliePath, 'smali')
  863. applicationFile = file_utils.getPackagePath(smaliPath, application)
  864. applicationFile += '.smali'
  865. ret = changeApplicationDex(applicationFile)
  866. if ret:
  867. return ret
  868. return splitSmali(game, sdk, subChannel, config, application)
  869. def changeApplicationDex(file):
  870. '''
  871. 修改application的smali文件,增加MultiDex操作
  872. '''
  873. index = file_utils.getApplicationSmaliIndex(file)
  874. file_utils.insertApplicationSmali(file, index)
  875. return 0
  876. def splitSmali(game, sdk, subChannel, config, application):
  877. '''
  878. 如果函数上限超过限制,自动拆分smali,以便生成多个dex文件
  879. '''
  880. print('splitSmali...')
  881. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  882. smaliPath = os.path.join(decompliePath, 'smali')
  883. appPackage = None
  884. if application:
  885. appPackage = application[:application.rfind('.')]
  886. appPackage = appPackage.replace('.', '/')
  887. allFiles = []
  888. allFiles = file_utils.list_files(smaliPath, allFiles)
  889. #print('file count is %d' % len(allFiles))
  890. #maxFuncNum = 65535
  891. # 留一点空间,防止计算误差
  892. maxFuncNum = 64000
  893. currFucNum = 0
  894. totalFucNum = 0
  895. currDexIndex = 1
  896. allRefs = []
  897. #保证Application等类在第一个classex.dex文件中
  898. for f in allFiles:
  899. f = f.replace('\\', '/')
  900. if (appPackage and appPackage in f) or '/android/support/multidex' in f:
  901. currFucNum += smali_utils.get_smali_method_count(f, allRefs)
  902. totalFucNum = currFucNum
  903. for f in allFiles:
  904. f = f.replace('\\', '/')
  905. if not f.endswith('.smali'):
  906. continue
  907. if (appPackage and appPackage in f) or '/android/support/multidex' in f:
  908. continue
  909. thisFucNum = smali_utils.get_smali_method_count(f, allRefs)
  910. totalFucNum += thisFucNum
  911. #print('%d # %d ==> %s' % (thisFucNum, currDexIndex, f))
  912. #print('totalFucNum is %d' % totalFucNum)
  913. if currFucNum + thisFucNum >= maxFuncNum:
  914. currFucNum = thisFucNum
  915. currDexIndex += 1
  916. newDexPath = os.path.join(decompliePath, 'smali_classes%d' % currDexIndex)
  917. os.makedirs(newDexPath)
  918. else:
  919. currFucNum += thisFucNum
  920. if currDexIndex > 1:
  921. newDexPath = os.path.join(decompliePath, 'smali_classes%d' % currDexIndex)
  922. targetFile = newDexPath + f[len(smaliPath):]
  923. file_utils.copyFile(f, targetFile, True)
  924. return 0
  925. def changeVersion(game, sdk, subChannel, config):
  926. '''
  927. 更改版本号
  928. '''
  929. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  930. yml = os.path.join(decompliePath, 'apktool.yml')
  931. versionCode = None
  932. versionName = None
  933. targetSdkVersion = None
  934. if 'versionCode' in config:
  935. versionCode = config['versionCode']
  936. if 'versionName' in config:
  937. versionName = config['versionName']
  938. if 'targetSdkVersion' in config:
  939. targetSdkVersion = config['targetSdkVersion']
  940. print('changeVersion versionCode=%s,versionName=%s,targetSdkVersion=%s' % (versionCode, versionName, targetSdkVersion))
  941. return file_utils.changeVersion(yml, versionCode, versionName, targetSdkVersion)
  942. def recomplie(game, sdk, subChannel, config):
  943. '''
  944. 回编译
  945. '''
  946. print('recomplie apk...')
  947. apktoolPath = file_utils.getApkToolPath()
  948. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  949. outApk = file_utils.getOutApkPath(game, sdk, subChannel, config['cache'])
  950. useAppt2 = ' --use-aapt2'
  951. if config['aapt2disable']:
  952. useAppt2 = ''
  953. return file_utils.execJarCmd(apktoolPath, 'b -f "%s" -o "%s"%s' % (decompliePath, outApk, useAppt2))
  954. def alignApk(game, sdk, subChannel, config):
  955. '''
  956. 对齐apk
  957. '''
  958. print('align apk...')
  959. outApk = file_utils.getOutApkPath(game, sdk, subChannel, config['cache'])
  960. alignApk = file_utils.getAlignApkPath(game, sdk, subChannel, config['cache'])
  961. alignapkTool = file_utils.getAlignPath()
  962. if os.path.exists(alignApk):
  963. os.remove(alignApk)
  964. ret = file_utils.getExecPermission(alignapkTool)
  965. if ret:
  966. return ret
  967. # zipalign.exe -v -p 4 input.apk output.apk
  968. return file_utils.execFormatCmd('"%s" -f -p 4 "%s" "%s"' % (alignapkTool, outApk, alignApk))
  969. def apksignerApk(game, sdk, subChannel, config):
  970. '''
  971. 签名apk
  972. '''
  973. print('sign apk...')
  974. path = os.path.join(file_utils.getCurrentPath(), 'keystore', 'key.json')
  975. jsonText = file_utils.readFile(path)
  976. signConfig = json.loads(jsonText)
  977. keystore = {}
  978. if game in signConfig:
  979. if sdk in signConfig[game] and subChannel in signConfig[game][sdk]:
  980. keystore = signConfig[game][sdk][subChannel]
  981. else:
  982. keystore = signConfig['default']
  983. else:
  984. keystore = signConfig['default']
  985. print('storeFile is "%s"' % keystore['storeFile'])
  986. apksigner = file_utils.getApksignerPath()
  987. alignApk = file_utils.getAlignApkPath(game, sdk, subChannel, config['cache'])
  988. signedApk = file_utils.getSignApkPath(game, sdk, subChannel, config['cache'])
  989. storeFile = os.path.join(file_utils.getCurrentPath(), 'keystore', keystore['storeFile'])
  990. if 'outName' in config and 'outPath' in config:
  991. if not os.path.exists(config['outPath']):
  992. os.makedirs(config['outPath'])
  993. signedApk = os.path.join(config['outPath'], config['outName'] + '.apk')
  994. elif 'outName' in config:
  995. signedApk = file_utils.getRenameApkPath(game, sdk, config['cache'], config['outName'])
  996. # java -jar apksigner.jar sign --ks key.jks --ks-key-alias releasekey --ks-pass pass:pp123456 --key-pass pass:pp123456 --out output.apk input.apk
  997. v2disable = ''
  998. if 'v2disable' in config and config['v2disable']:
  999. v2disable = ' --v2-signing-enabled=false'
  1000. return file_utils.execJarCmd(apksigner, 'sign%s --ks "%s" --ks-key-alias %s --ks-pass pass:%s --key-pass pass:%s --out "%s" "%s"' % (v2disable, storeFile, keystore['keyAlias'], keystore['storePassword'], keystore['keyPassword'], signedApk, alignApk))
  1001. def addChannel(game, sdk, subChannel, config):
  1002. '''
  1003. 添加渠道信息
  1004. '''
  1005. if 'v2disable' in config and config['v2disable']:
  1006. return 0
  1007. walle = file_utils.getWallePath()
  1008. signedApk = file_utils.getSignApkPath(game, sdk, subChannel, config['cache'])
  1009. if 'outName' in config and 'outPath' in config:
  1010. signedApk = os.path.join(config['outPath'], config['outName'] + '.apk')
  1011. elif 'outName' in config:
  1012. signedApk = file_utils.getRenameApkPath(game, sdk, config['cache'], config['outName'])
  1013. properties = config['properties']
  1014. appid = ''
  1015. appkey = ''
  1016. host = ''
  1017. if 'appid' in properties:
  1018. appid = properties['appid']
  1019. if 'appkey' in properties:
  1020. appkey = properties['appkey']
  1021. if 'host' in properties:
  1022. host = properties['host']
  1023. return file_utils.execJarCmd(walle, 'put -e version=%s,agent=%s,appid=%s,appkey=%s,host=%s "%s" "%s"' % (config_utils.getDate(), properties['agent'], appid, appkey, host, signedApk, signedApk))
  1024. def clearTemp(game, sdk, subChannel, config):
  1025. '''
  1026. 清空中间产生的文件
  1027. '''
  1028. print('clear temp...')
  1029. alignApk = file_utils.getAlignApkPath(game, sdk, subChannel, config['cache'])
  1030. outApk = file_utils.getOutApkPath(game, sdk, subChannel, config['cache'])
  1031. if os.path.exists(alignApk):
  1032. os.remove(alignApk)
  1033. if os.path.exists(outApk):
  1034. os.remove(outApk)
  1035. decompliePath = file_utils.getDecompliePath(game, sdk, subChannel, config['cache'])
  1036. file_utils.deleteFolder(decompliePath)
  1037. print('clear temp end')
  1038. def packConsoleInput():
  1039. '''
  1040. 控制台打包
  1041. '''
  1042. if len(sys.argv) < 3:
  1043. print('argument is missing')
  1044. return 1
  1045. # 校验参数
  1046. game = sys.argv[1]
  1047. sdk = sys.argv[2]
  1048. # 可选参数,没有则默认打全部渠道
  1049. subChannel = None
  1050. if len(sys.argv) > 3:
  1051. subChannel = sys.argv[3]
  1052. return packConsole(game, sdk, subChannel)
  1053. def packConsole(game, sdk, subChannel):
  1054. '''
  1055. 控制台打包
  1056. '''
  1057. if not os.path.exists(file_utils.getFullGameApk(game)):
  1058. print('game "%s" not exists' % game)
  1059. return 1
  1060. if not os.path.exists(file_utils.getFullSDKPath(sdk)):
  1061. print('sdk "%s" not exists' % sdk)
  1062. return 1
  1063. # 读取配置
  1064. channelPath = file_utils.getChannelPath(game, sdk)
  1065. configPath = os.path.join(channelPath, 'config.json')
  1066. if not os.path.exists(configPath):
  1067. print('%s not exists' % configPath)
  1068. return 1
  1069. jsonText = file_utils.readFile(configPath)
  1070. config = json.loads(jsonText)
  1071. # 检查参数
  1072. if not config_utils.checkConfig(config):
  1073. return 1
  1074. # 处理参数
  1075. config_utils.replaceArgs(config)
  1076. successCount = 0
  1077. failureCount = 0
  1078. if type(config) == dict:
  1079. if subChannel is None or config['subChannel'] == subChannel:
  1080. ret = pack(game, sdk, config)
  1081. if ret:
  1082. failureCount += 1
  1083. else:
  1084. successCount += 1
  1085. else:
  1086. print('subChannel "%s" no found' % subChannel)
  1087. return 1
  1088. elif type(config) == list:
  1089. found = False
  1090. for itemConfig in config:
  1091. if subChannel is None or itemConfig['subChannel'] == subChannel:
  1092. found = True
  1093. ret = pack(game, sdk, itemConfig)
  1094. if ret:
  1095. failureCount += 1
  1096. else:
  1097. successCount += 1
  1098. if not found:
  1099. print('subChannel "%s" no found' % subChannel)
  1100. return 1
  1101. print('success %d, failure %d' % (successCount, failureCount))
  1102. return 0