axmol/tools/jenkins-scripts/pull-request-builder.py

308 lines
11 KiB
Python
Raw Normal View History

2014-10-14 17:31:53 +08:00
#Github pull reqest builder for Jenkins
import json
import os
import re
import urllib2
import urllib
import base64
import requests
2014-01-02 10:19:48 +08:00
import sys
2014-01-02 17:09:49 +08:00
import traceback
2014-01-06 20:58:00 +08:00
import platform
import subprocess
import codecs
from shutil import copy
import MySQLdb
#set Jenkins build description using submitDescription to mock browser behavior
#TODO: need to set parent build description
2014-01-06 20:58:00 +08:00
def set_description(desc, url):
req_data = urllib.urlencode({'description': desc})
2014-01-06 20:58:00 +08:00
req = urllib2.Request(url + 'submitDescription', req_data)
#print(os.environ['BUILD_URL'])
2014-10-14 17:31:53 +08:00
req.add_header('Content-Type', 'application/x-www-form-urlencoded')
2015-03-03 16:36:42 +08:00
base64string = base64.encodestring(os.environ['JENKINS_ADMIN'] + ":" + os.environ['JENKINS_ADMIN_PW']).replace('\n', '')
req.add_header("Authorization", "Basic " + base64string)
2014-01-02 17:09:49 +08:00
try:
urllib2.urlopen(req)
except:
traceback.print_exc()
def check_current_3rd_libs(branch):
2014-09-24 12:57:47 +08:00
print("start backup old 3rd libs...")
#get current_libs config
backup_files = range(2)
current_files = range(2)
2015-03-03 16:36:42 +08:00
config_file_paths = ['external/config.json', 'templates/lua-template-runtime/runtime/config.json']
for i, config_file_path in enumerate(config_file_paths):
if not os.path.isfile(config_file_path):
raise Exception("Could not find 'external/config.json'")
with open(config_file_path) as data_file:
data = json.load(data_file)
current_3rd_libs_version = data["version"]
filename = current_3rd_libs_version + '.zip'
node_name = os.environ['NODE_NAME']
backup_file = '../../../cocos-2dx-external/node/' + node_name + '/' + filename
backup_files[i] = backup_file
current_file = filename
current_files[i] = current_file
if os.path.isfile(backup_file):
2015-03-03 16:36:42 +08:00
copy(backup_file, current_file)
#run download-deps.py
2014-09-24 12:53:30 +08:00
print("prepare to downloading ...")
os.system('python download-deps.py -r no')
#backup file
for i, backup_file in enumerate(backup_files):
current_file = current_files[i]
copy(current_file, backup_file)
def connect_db():
db_host = os.environ['db_host']
db_user = os.environ['db_user']
db_pw = os.environ['db_pw']
2015-03-03 16:36:42 +08:00
db_name = os.environ['db_name']
2014-11-26 09:52:27 +08:00
try:
db = MySQLdb.connect(db_host, db_user, db_pw, db_name)
except:
traceback.print_exc()
return db
def close_db(db):
2014-11-26 09:52:27 +08:00
try:
db.close()
except:
traceback.print_exc()
def save_build_stats(db, pr, filename, size):
2014-11-26 09:52:27 +08:00
try:
cursor = db.cursor()
sql = "INSERT INTO `%s` (number, size, createdTime) VALUES(%d, %d, now())" % (filename, pr, size)
print sql
cursor.execute(sql)
db.commit()
except:
traceback.print_exc()
def scan_all_libs(db, pr_num):
stats = {}
lib_path = './tests/cpp-tests/proj.android/obj/local/armeabi'
for root, dirs, files in os.walk(lib_path):
2015-03-03 16:36:42 +08:00
for _file in files:
if not _file.endswith(".a"):
continue
print _file
libfile = lib_path + '/' + _file
_filename = _file.split('.')[0]
filesize = os.path.getsize(libfile) / 1024
stats[_filename] = filesize
save_build_stats(db, pr_num, _filename, filesize)
return stats
http_proxy = ''
2015-03-03 16:36:42 +08:00
if('HTTP_PROXY' in os.environ):
http_proxy = os.environ['HTTP_PROXY']
2015-03-03 16:36:42 +08:00
proxyDict = {'http': http_proxy, 'https': http_proxy}
2014-01-02 17:09:49 +08:00
def main():
#get payload from os env
payload_str = os.environ['payload']
2015-03-03 16:36:42 +08:00
payload_str = payload_str.decode('utf-8', 'ignore')
2014-01-02 17:09:49 +08:00
#parse to json obj
payload = json.loads(payload_str)
#get pull number
pr_num = payload['number']
print 'pr_num:' + str(pr_num)
#build for pull request action 'open' and 'synchronize', skip 'close'
action = payload['action']
print 'action: ' + action
2014-01-06 20:58:00 +08:00
#pr = payload['pull_request']
2014-10-14 17:31:53 +08:00
url = payload['html_url']
2014-01-02 17:09:49 +08:00
print "url:" + url
2015-03-03 16:36:42 +08:00
pr_desc = '<h3><a href=' + url + '> pr#' + str(pr_num) + ' is ' + action + '</a></h3>'
2014-01-02 17:09:49 +08:00
#get statuses url
statuses_url = payload['statuses_url']
2014-01-02 17:09:49 +08:00
#get pr target branch
branch = payload['branch']
2014-01-02 17:09:49 +08:00
#set commit status to pending
2014-01-23 00:21:58 +08:00
#target_url = os.environ['BUILD_URL']
jenkins_url = os.environ['JENKINS_URL']
job_name = os.environ['JOB_NAME'].split('/')[0]
build_number = os.environ['BUILD_NUMBER']
target_url = jenkins_url + 'job/' + job_name + '/' + build_number + '/'
2014-01-06 20:58:00 +08:00
set_description(pr_desc, target_url)
2014-10-14 17:31:53 +08:00
2015-03-03 16:36:42 +08:00
data = {"state": "pending", "target_url": target_url, "context": "Jenkins CI", "description": "Build started..."}
access_token = os.environ['GITHUB_ACCESS_TOKEN']
2015-03-03 16:36:42 +08:00
Headers = {"Authorization": "token " + access_token}
2014-01-02 17:09:49 +08:00
try:
2015-03-03 16:36:42 +08:00
requests.post(statuses_url, data=json.dumps(data), headers=Headers, proxies=proxyDict)
2014-01-02 17:09:49 +08:00
except:
traceback.print_exc()
2014-01-02 17:09:49 +08:00
#reset path to workspace root
2015-03-03 16:36:42 +08:00
os.system("cd " + os.environ['WORKSPACE'])
#pull latest code
2015-03-03 16:36:42 +08:00
os.system("git pull origin " + branch)
os.system("git checkout " + branch)
2014-01-07 15:41:18 +08:00
os.system("git branch -D pull" + str(pr_num))
#clean workspace
2014-10-14 17:31:53 +08:00
print "Before checkout: git clean -xdf -f"
os.system("git clean -xdf -f")
#fetch pull request to local repo
git_fetch_pr = "git fetch origin pull/" + str(pr_num) + "/head"
ret = os.system(git_fetch_pr)
if(ret != 0):
return(2)
2014-10-14 17:31:53 +08:00
2015-03-03 16:36:42 +08:00
#checkout a new branch from v3 or v4-develop
git_checkout = "git checkout -b " + "pull" + str(pr_num)
os.system(git_checkout)
#merge pull reqeust head
p = os.popen('git merge --no-edit FETCH_HEAD')
r = p.read()
#check if merge fail
if r.find('CONFLICT') > 0:
print r
return(3)
2014-10-14 17:31:53 +08:00
# After checkout a new branch, clean workspace again
2014-10-14 17:31:53 +08:00
print "After checkout: git clean -xdf -f"
os.system("git clean -xdf -f")
#update submodule
git_update_submodule = "git submodule update --init --force"
ret = os.system(git_update_submodule)
if(ret != 0):
return(2)
2014-01-06 20:58:00 +08:00
#copy check_current_3rd_libs
2014-09-24 12:42:53 +08:00
check_current_3rd_libs(branch)
2014-01-08 16:21:11 +08:00
# Generate binding glue codes
2015-03-03 16:36:42 +08:00
if(branch == 'v3' or branch == 'v4-develop'):
ret = os.system("python tools/jenkins-scripts/gen_jsb.py")
if(ret != 0):
return(1)
2014-01-08 16:21:11 +08:00
2014-01-07 15:41:18 +08:00
#make temp dir
print "current dir is: " + os.environ['WORKSPACE']
2015-03-03 16:36:42 +08:00
os.system("cd " + os.environ['WORKSPACE'])
2014-01-07 15:41:18 +08:00
os.mkdir("android_build_objs")
2014-01-06 20:58:00 +08:00
#add symbol link
2015-03-03 16:36:42 +08:00
PROJECTS = ["cpp-empty-test", "cpp-tests"]
print platform.system()
if(platform.system() == 'Darwin'):
for item in PROJECTS:
2015-03-03 16:36:42 +08:00
cmd = "ln -s " + os.environ['WORKSPACE'] + "/android_build_objs/ " + os.environ['WORKSPACE'] + "/tests/" + item + "/proj.android/obj"
os.system(cmd)
elif(platform.system() == 'Windows'):
for item in PROJECTS:
2015-03-03 16:36:42 +08:00
p = item.replace("/", os.sep)
cmd = "mklink /J " + os.environ['WORKSPACE'] + os.sep + "tests" + os.sep + p + os.sep + "proj.android" + os.sep + "obj " + os.environ['WORKSPACE'] + os.sep + "android_build_objs"
print cmd
os.system(cmd)
2014-10-14 17:31:53 +08:00
2014-01-02 17:09:49 +08:00
#build
2015-03-03 16:36:42 +08:00
# TODO: add wp8.1 universal build
2014-01-02 17:09:49 +08:00
#TODO: add mac build
node_name = os.environ['NODE_NAME']
2015-03-03 16:36:42 +08:00
jenkins_script_path = 'tools/jenkins-scripts/'
if(branch == 'v3' or branch == 'v4-develop'):
if(node_name == 'android') or (node_name == 'android_bak'):
2015-03-03 16:36:42 +08:00
#modify tests/cpp-empty-test/Classes/AppDelegate.cpp to support Console
# FIXME: We should use patch instead
modify_file = 'tests/cpp-empty-test/Classes/AppDelegate.cpp'
data = codecs.open(modify_file, encoding='UTF-8').read()
data = re.sub("director->setDisplayStats\(true\);", "director->setDisplayStats(true); director->getConsole()->listenOnTCP(5678);", data)
codecs.open(modify_file, 'wb', encoding='UTF-8').write(data)
#modify tests/cpp-empty-test/proj.android/AndroidManifest.xml to support Console
modify_file = 'tests/cpp-empty-test/proj.android/AndroidManifest.xml'
data = codecs.open(modify_file, encoding='UTF-8').read()
data = re.sub('<uses-feature android:glEsVersion="0x00020000" />', '<uses-feature android:glEsVersion="0x00020000" /> <uses-permission android:name="android.permission.INTERNET"/>', data)
codecs.open(modify_file, 'wb', encoding='UTF-8').write(data)
print "Start build android..."
ret = os.system("python build/android-build.py -p 10 all")
# create and save apk
if(ret == 0):
sample_dir = 'tests/cpp-tests/proj.android/'
local_apk = sample_dir + 'bin/CppTests-debug.apk'
backup_apk = os.environ['BACKUP_PATH'] + 'CppTests_' + str(pr_num) + '.apk'
os.system('cp ' + local_apk + ' ' + backup_apk)
db = connect_db()
scan_all_libs(db, pr_num)
ret = os.system("python build/android-build.py -p 10 -b release cpp-empty-test")
if(ret == 0):
_path = 'tests/cpp-empty-test/proj.android/libs/armeabi/libcpp_empty_test.so'
filesize = os.path.getsize(_path)
pr_desc = pr_desc + '<h3>size of libcpp_empty_test.so is:' + str(filesize / 1024) + 'kb</h3>'
set_description(pr_desc, target_url)
save_build_stats(db, pr_num, 'libcpp_empty_test', filesize / 1024)
ret = os.system("python build/android-build.py -p 10 -b release lua-empty-test")
if(ret == 0):
_path = 'tests/lua-empty-test/project/proj.android/libs/armeabi/liblua_empty_test.so'
filesize = os.path.getsize(_path)
pr_desc = pr_desc + '<h3>size of liblua_empty_test.so is:' + str(filesize / 1024) + 'kb</h3>'
set_description(pr_desc, target_url)
save_build_stats(db, pr_num, 'liblua_empty_test', filesize / 1024)
close_db(db)
elif(node_name == 'win32' or node_name == 'win32_win7' or node_name == 'win32_bak'):
2015-03-04 14:12:11 +08:00
ret = subprocess.call('"%VS120COMNTOOLS%..\IDE\devenv.com" "build\cocos2d-win32.vc2012.sln" /Build "Debug|Win32"', shell=True)
elif(node_name == 'windows-universal' or node_name == 'windows-universal_bak'):
2015-03-03 16:36:42 +08:00
ret = subprocess.call('"%VS120COMNTOOLS%..\IDE\devenv.com" "build\cocos2d-win8.1-universal.sln" /Build "Debug|Win32"', shell=True)
elif(node_name == 'ios_mac' or node_name == 'ios' or node_name == 'ios_bak'):
2015-03-03 16:36:42 +08:00
ret = os.system(jenkins_script_path + "ios-build.sh")
elif(node_name == 'mac' or node_name == 'mac_bak'):
2015-03-03 16:36:42 +08:00
ret = os.system(jenkins_script_path + "mac-build.sh")
elif(node_name == 'linux_centos' or node_name == 'linux' or node_name == 'linux_bak'):
2015-03-03 16:36:42 +08:00
ret = os.system(jenkins_script_path + "linux-build.sh")
2014-01-02 17:09:49 +08:00
#get build result
print "build finished and return " + str(ret)
2014-10-14 17:31:53 +08:00
exit_code = 1
2014-01-02 17:09:49 +08:00
if ret == 0:
exit_code = 0
else:
exit_code = 1
2014-10-14 17:31:53 +08:00
#clean workspace
os.system("cd " + os.environ['WORKSPACE'])
os.system("git reset --hard")
os.system("git clean -xdf -f")
2015-03-03 16:36:42 +08:00
os.system("git checkout " + branch)
os.system("git branch -D pull" + str(pr_num))
2014-01-06 20:58:00 +08:00
return(exit_code)
2014-01-02 17:09:49 +08:00
# -------------- main --------------
if __name__ == '__main__':
2014-01-06 20:58:00 +08:00
sys_ret = 0
2014-10-14 17:31:53 +08:00
try:
2014-01-06 20:58:00 +08:00
sys_ret = main()
2014-01-02 17:09:49 +08:00
except:
traceback.print_exc()
2014-01-06 20:58:00 +08:00
sys_ret = 1
finally:
sys.exit(sys_ret)