2014-10-14 17:31:53 +08:00
|
|
|
#Github pull reqest builder for Jenkins
|
2013-12-30 21:11:32 +08:00
|
|
|
|
|
|
|
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
|
2014-01-24 14:40:15 +08:00
|
|
|
import subprocess
|
2014-03-20 16:19:35 +08:00
|
|
|
import codecs
|
2014-05-07 19:01:46 +08:00
|
|
|
from shutil import copy
|
2014-07-17 14:52:26 +08:00
|
|
|
import MySQLdb
|
2013-12-30 21:11:32 +08:00
|
|
|
|
|
|
|
#set Jenkins build description using submitDescription to mock browser behavior
|
2014-05-07 19:01:46 +08:00
|
|
|
#TODO: need to set parent build description
|
|
|
|
|
2014-01-06 20:58:00 +08:00
|
|
|
def set_description(desc, url):
|
2013-12-30 21:11:32 +08:00
|
|
|
req_data = urllib.urlencode({'description': desc})
|
2014-01-06 20:58:00 +08:00
|
|
|
req = urllib2.Request(url + 'submitDescription', req_data)
|
2013-12-31 18:29:32 +08:00
|
|
|
#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', '')
|
2013-12-30 21:11:32 +08:00
|
|
|
req.add_header("Authorization", "Basic " + base64string)
|
2014-01-02 17:09:49 +08:00
|
|
|
try:
|
|
|
|
urllib2.urlopen(req)
|
|
|
|
except:
|
2014-01-04 16:25:31 +08:00
|
|
|
traceback.print_exc()
|
2014-05-07 19:01:46 +08:00
|
|
|
|
2014-05-14 11:19:30 +08:00
|
|
|
def check_current_3rd_libs(branch):
|
2014-09-24 12:57:47 +08:00
|
|
|
print("start backup old 3rd libs...")
|
2014-05-07 19:01:46 +08:00
|
|
|
#get current_libs config
|
2014-05-12 18:49:52 +08:00
|
|
|
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']
|
|
|
|
|
2014-05-12 18:49:52 +08:00
|
|
|
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'")
|
2014-05-07 19:01:46 +08:00
|
|
|
|
2014-05-12 18:49:52 +08:00
|
|
|
with open(config_file_path) as data_file:
|
|
|
|
data = json.load(data_file)
|
2014-05-07 19:01:46 +08:00
|
|
|
|
2014-05-12 18:49:52 +08:00
|
|
|
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)
|
2014-05-12 18:49:52 +08:00
|
|
|
#run download-deps.py
|
2014-09-24 12:53:30 +08:00
|
|
|
print("prepare to downloading ...")
|
2014-05-07 20:34:06 +08:00
|
|
|
os.system('python download-deps.py -r no')
|
|
|
|
#backup file
|
2014-05-12 18:49:52 +08:00
|
|
|
for i, backup_file in enumerate(backup_files):
|
|
|
|
current_file = current_files[i]
|
|
|
|
copy(current_file, backup_file)
|
2014-05-07 19:01:46 +08:00
|
|
|
|
2014-08-07 17:50:56 +08:00
|
|
|
def connect_db():
|
2014-07-17 14:52:26 +08:00
|
|
|
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()
|
2014-08-07 17:50:56 +08:00
|
|
|
return db
|
|
|
|
|
|
|
|
def close_db(db):
|
2014-11-26 09:52:27 +08:00
|
|
|
try:
|
|
|
|
db.close()
|
|
|
|
except:
|
|
|
|
traceback.print_exc()
|
2014-08-07 17:50:56 +08:00
|
|
|
|
|
|
|
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()
|
2014-08-07 17:50:56 +08:00
|
|
|
|
|
|
|
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)
|
2014-08-07 17:50:56 +08:00
|
|
|
return stats
|
2014-07-17 14:52:26 +08:00
|
|
|
|
2014-05-30 12:51:54 +08:00
|
|
|
http_proxy = ''
|
2015-03-03 16:36:42 +08:00
|
|
|
|
|
|
|
if('HTTP_PROXY' in os.environ):
|
2014-05-30 12:51:54 +08:00
|
|
|
http_proxy = os.environ['HTTP_PROXY']
|
2015-03-03 16:36:42 +08:00
|
|
|
proxyDict = {'http': http_proxy, 'https': http_proxy}
|
2014-07-17 14:52:26 +08:00
|
|
|
|
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
|
|
|
|
2014-02-07 14:57:07 +08:00
|
|
|
#pr = payload['pull_request']
|
2014-10-14 17:31:53 +08:00
|
|
|
|
2014-02-07 14:57:07 +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>'
|
2013-12-30 21:11:32 +08:00
|
|
|
|
2014-01-02 17:09:49 +08:00
|
|
|
#get statuses url
|
2014-02-07 14:57:07 +08:00
|
|
|
statuses_url = payload['statuses_url']
|
2014-01-02 17:09:49 +08:00
|
|
|
|
|
|
|
#get pr target branch
|
2014-02-07 14:57:07 +08:00
|
|
|
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']
|
2014-01-24 12:05:31 +08:00
|
|
|
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..."}
|
2014-01-04 16:25:31 +08:00
|
|
|
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:
|
2014-01-04 16:25:31 +08:00
|
|
|
traceback.print_exc()
|
2014-01-02 17:09:49 +08:00
|
|
|
|
2014-01-03 10:59:37 +08:00
|
|
|
#reset path to workspace root
|
2015-03-03 16:36:42 +08:00
|
|
|
os.system("cd " + os.environ['WORKSPACE'])
|
2014-07-25 16:38:22 +08:00
|
|
|
#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"
|
2014-03-11 21:39:31 +08:00
|
|
|
os.system("git clean -xdf -f")
|
2014-01-03 10:59:37 +08:00
|
|
|
#fetch pull request to local repo
|
2014-05-15 10:36:01 +08:00
|
|
|
git_fetch_pr = "git fetch origin pull/" + str(pr_num) + "/head"
|
2014-03-26 14:29:53 +08:00
|
|
|
ret = os.system(git_fetch_pr)
|
|
|
|
if(ret != 0):
|
2014-04-08 16:35:01 +08:00
|
|
|
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
|
2014-07-25 16:38:22 +08:00
|
|
|
git_checkout = "git checkout -b " + "pull" + str(pr_num)
|
2014-01-03 10:59:37 +08:00
|
|
|
os.system(git_checkout)
|
2014-07-25 16:38:22 +08:00
|
|
|
#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
|
|
|
|
2014-03-11 22:05:50 +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"
|
2014-03-11 22:05:50 +08:00
|
|
|
os.system("git clean -xdf -f")
|
|
|
|
|
2014-01-03 10:59:37 +08:00
|
|
|
#update submodule
|
|
|
|
git_update_submodule = "git submodule update --init --force"
|
2014-03-18 14:37:23 +08:00
|
|
|
ret = os.system(git_update_submodule)
|
|
|
|
if(ret != 0):
|
2014-04-08 16:35:01 +08:00
|
|
|
return(2)
|
2014-01-06 20:58:00 +08:00
|
|
|
|
2014-05-07 19:01:46 +08:00
|
|
|
#copy check_current_3rd_libs
|
2014-09-24 12:42:53 +08:00
|
|
|
check_current_3rd_libs(branch)
|
2014-05-07 19:01:46 +08:00
|
|
|
|
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")
|
|
|
|
|
2014-03-27 14:28:40 +08:00
|
|
|
if(ret != 0):
|
|
|
|
return(1)
|
2014-01-08 16:21:11 +08:00
|
|
|
|
2014-01-07 15:41:18 +08:00
|
|
|
#make temp dir
|
2014-03-11 21:39:31 +08:00
|
|
|
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"]
|
2014-03-11 19:39:20 +08:00
|
|
|
|
|
|
|
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)
|
2014-03-11 19:39:20 +08:00
|
|
|
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
|
2014-01-24 14:40:15 +08:00
|
|
|
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'):
|
2015-03-03 16:42:48 +08:00
|
|
|
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)
|
2015-03-03 16:42:48 +08:00
|
|
|
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)
|
2015-03-03 16:42:48 +08:00
|
|
|
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)
|
2015-03-03 16:42:48 +08:00
|
|
|
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")
|
2015-03-03 16:42:48 +08:00
|
|
|
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")
|
2015-03-03 16:42:48 +08:00
|
|
|
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
|
|
|
|
2014-01-04 16:25:31 +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
|
|
|
|
2013-12-31 18:29:32 +08:00
|
|
|
#clean workspace
|
2014-03-21 13:57:38 +08:00
|
|
|
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)
|
2013-12-31 18:29:32 +08:00
|
|
|
os.system("git branch -D pull" + str(pr_num))
|
2013-12-30 21:11:32 +08:00
|
|
|
|
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:
|
2014-01-04 16:25:31 +08:00
|
|
|
traceback.print_exc()
|
2014-01-06 20:58:00 +08:00
|
|
|
sys_ret = 1
|
|
|
|
finally:
|
|
|
|
sys.exit(sys_ret)
|