Jenkins pipeline to create master-ci (#1019)
* Added Jenkinsfile * Added Jenkinsfile * Added Jenkinsfile * change order * sudo * whoami? * Added Jenkinsfile * install git * Untested build scripts * Add lockable resource * Fix syntax * Only one stage * fix target dir * Use deploy key * noqa on test_openpilot * Fix version.h path * Cleanup release files * Add linter scripts to release * Update jenkinsfile * Fix path * this should work * Use python3 docker container * Run in correct directory * Setup /data/pythonpath Co-authored-by: commaci-public <60409688+commaci-public@users.noreply.github.com>pull/1026/head
parent
f81a43381f
commit
0319861700
|
@ -0,0 +1,23 @@
|
|||
pipeline {
|
||||
agent {
|
||||
docker {
|
||||
image 'python:3.7.3'
|
||||
args '--user=root'
|
||||
}
|
||||
|
||||
}
|
||||
stages {
|
||||
stage('EON Build/Test') {
|
||||
steps {
|
||||
lock(resource: "", label: 'eon', inversePrecedence: true, variable: 'eon_name', quantity: 1){
|
||||
timeout(time: 30, unit: 'MINUTES') {
|
||||
dir(path: 'release') {
|
||||
sh 'pip install paramiko'
|
||||
sh 'python remote_build.py'
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
9
Makefile
9
Makefile
|
@ -1,9 +0,0 @@
|
|||
|
||||
code_dir := $(shell pwd)
|
||||
|
||||
# TODO: Add a global build system
|
||||
|
||||
.PHONY: all
|
||||
all:
|
||||
cd selfdrive && PYTHONPATH=$(code_dir) PREPAREONLY=1 ./manager.py
|
||||
|
|
@ -1,28 +1,19 @@
|
|||
README.md
|
||||
|
||||
launch_openpilot.sh
|
||||
SAFETY.md
|
||||
|
||||
.travis.yml
|
||||
.pylintrc
|
||||
run_docker_tests.sh
|
||||
Dockerfile.openpilot
|
||||
phonelibs/install_capnp.sh
|
||||
check_code_quality.sh
|
||||
|
||||
Pipfile
|
||||
Pipfile.lock
|
||||
|
||||
.gitignore
|
||||
LICENSE
|
||||
Makefile
|
||||
launch_chffrplus.sh
|
||||
launch_openpilot.sh
|
||||
|
||||
CONTRIBUTING.md
|
||||
RELEASES.md
|
||||
|
||||
SConstruct
|
||||
|
||||
flake8_openpilot.sh
|
||||
pylint_openpilot.sh
|
||||
|
||||
.github/**
|
||||
|
||||
apk/ai.comma*.apk
|
||||
|
@ -312,7 +303,6 @@ selfdrive/test/process_replay/update_refs.py
|
|||
selfdrive/test/process_replay/ref_commit
|
||||
selfdrive/test/process_replay/README.md
|
||||
|
||||
selfdrive/ui/.gitignore
|
||||
selfdrive/ui/SConscript
|
||||
selfdrive/ui/*.c
|
||||
selfdrive/ui/*.cc
|
||||
|
|
|
@ -0,0 +1,120 @@
|
|||
#!/data/data/com.termux/files/usr/bin/bash -e
|
||||
|
||||
mkdir -p /dev/shm
|
||||
chmod 777 /dev/shm
|
||||
|
||||
|
||||
add_subtree() {
|
||||
echo "[-] adding $2 subtree T=$SECONDS"
|
||||
if [ -d "$2" ]; then
|
||||
if git subtree pull --prefix "$2" https://github.com/commaai/"$1".git "$3" --squash -m "Merge $2 subtree"; then
|
||||
echo "git subtree pull succeeds"
|
||||
else
|
||||
echo "git subtree pull failed, fixing"
|
||||
git merge --abort || true
|
||||
git rm -r $2
|
||||
git commit -m "Remove old $2 subtree"
|
||||
git subtree add --prefix "$2" https://github.com/commaai/"$1".git "$3" --squash
|
||||
fi
|
||||
else
|
||||
git subtree add --prefix "$2" https://github.com/commaai/"$1".git "$3" --squash
|
||||
fi
|
||||
}
|
||||
|
||||
SOURCE_DIR=/data/openpilot_source
|
||||
TARGET_DIR=/data/openpilot
|
||||
|
||||
ln -sf $TARGET_DIR /data/pythonpath
|
||||
|
||||
export GIT_COMMITTER_NAME="Vehicle Researcher"
|
||||
export GIT_COMMITTER_EMAIL="user@comma.ai"
|
||||
export GIT_AUTHOR_NAME="Vehicle Researcher"
|
||||
export GIT_AUTHOR_EMAIL="user@comma.ai"
|
||||
export GIT_SSH_COMMAND="ssh -i /tmp/deploy_key"
|
||||
|
||||
echo "[-] Setting up repo T=$SECONDS"
|
||||
if [ ! -d "$TARGET_DIR" ]; then
|
||||
mkdir -p $TARGET_DIR
|
||||
cd $TARGET_DIR
|
||||
git init
|
||||
git remote add origin git@github.com:commaai/openpilot.git
|
||||
fi
|
||||
|
||||
|
||||
echo "[-] fetching public T=$SECONDS"
|
||||
cd $TARGET_DIR
|
||||
git prune || true
|
||||
git remote prune origin || true
|
||||
|
||||
echo "[-] bringing master-ci and devel in sync T=$SECONDS"
|
||||
git fetch origin master-ci
|
||||
git fetch origin devel
|
||||
|
||||
git checkout --track origin/master-ci || true
|
||||
git reset --hard master-ci
|
||||
git checkout master-ci
|
||||
git reset --hard origin/devel
|
||||
git clean -xdf
|
||||
|
||||
# subtrees to make updates more reliable. updating them needs a clean tree
|
||||
add_subtree "cereal" "cereal" master
|
||||
add_subtree "panda" "panda" master
|
||||
add_subtree "opendbc" "opendbc" master
|
||||
add_subtree "openpilot-pyextra" "pyextra" master
|
||||
|
||||
# leave .git alone
|
||||
echo "[-] erasing old openpilot T=$SECONDS"
|
||||
rm -rf $TARGET_DIR/* $TARGET_DIR/.gitmodules
|
||||
|
||||
# dont delete our subtrees
|
||||
git checkout -- cereal panda opendbc pyextra
|
||||
|
||||
# reset tree and get version
|
||||
cd $SOURCE_DIR
|
||||
git clean -xdf
|
||||
git checkout -- selfdrive/common/version.h
|
||||
|
||||
VERSION=$(cat selfdrive/common/version.h | awk -F\" '{print $2}')
|
||||
echo "#define COMMA_VERSION \"$VERSION-release\"" > selfdrive/common/version.h
|
||||
|
||||
# do the files copy
|
||||
echo "[-] copying files T=$SECONDS"
|
||||
cd $SOURCE_DIR
|
||||
cp -pR --parents $(cat release/files_common) $TARGET_DIR/
|
||||
|
||||
# in the directory
|
||||
cd $TARGET_DIR
|
||||
|
||||
echo "[-] committing version $VERSION T=$SECONDS"
|
||||
git add -f .
|
||||
git status
|
||||
git commit -a -m "openpilot v$VERSION release"
|
||||
|
||||
# Run build
|
||||
SCONS_CACHE=1 scons -j3
|
||||
|
||||
echo "[-] testing openpilot T=$SECONDS"
|
||||
PYTHONPATH="$SOURCE_DIR:$SOURCE_DIR/pyextra" nosetests -s selfdrive/test/test_openpilot.py
|
||||
|
||||
echo "[-] testing panda build T=$SECONDS"
|
||||
pushd panda/board/
|
||||
make bin
|
||||
popd
|
||||
|
||||
echo "[-] testing pedal build T=$SECONDS"
|
||||
pushd panda/board/pedal
|
||||
make obj/comma.bin
|
||||
popd
|
||||
|
||||
if [ ! -z "$PUSH" ]; then
|
||||
echo "[-] Pushing to $PUSH T=$SECONDS"
|
||||
git push -f origin master:$PUSH
|
||||
fi
|
||||
|
||||
echo "[-] done pushing T=$SECONDS"
|
||||
|
||||
# reset version
|
||||
cd $SOURCE_DIR
|
||||
git checkout -- selfdrive/common/version.h
|
||||
|
||||
echo "[-] done T=$SECONDS"
|
|
@ -1,4 +1,4 @@
|
|||
-----BEGIN PRIVATE KEY-----
|
||||
-----BEGIN RSA PRIVATE KEY-----
|
||||
MIIEvAIBADANBgkqhkiG9w0BAQEFAASCBKYwggSiAgEAAoIBAQC+iXXq30Tq+J5N
|
||||
Kat3KWHCzcmwZ55nGh6WggAqECa5CasBlM9VeROpVu3beA+5h0MibRgbD4DMtVXB
|
||||
t6gEvZ8nd04E7eLA9LTZyFDZ7SkSOVj4oXOQsT0GnJmKrASW5KslTWqVzTfo2XCt
|
||||
|
@ -25,4 +25,4 @@ jTadcgKFnRUmc+JT9p/ZbCxkA/ALFg8++G+0ghECgYA8vG3M/utweLvq4RI7l7U7
|
|||
b+i2BajfK2OmzNi/xugfeLjY6k2tfQGRuv6ppTjehtji2uvgDWkgjJUgPfZpir3I
|
||||
RsVMUiFgloWGHETOy0Qvc5AwtqTJFLTD1Wza2uBilSVIEsg6Y83Gickh+ejOmEsY
|
||||
6co17RFaAZHwGfCFFjO76Q==
|
||||
-----END PRIVATE KEY-----
|
||||
-----END RSA PRIVATE KEY-----
|
||||
|
|
|
@ -0,0 +1,73 @@
|
|||
#!/usr/bin/env python2
|
||||
import paramiko
|
||||
import os
|
||||
import sys
|
||||
import re
|
||||
import time
|
||||
import socket
|
||||
|
||||
|
||||
def start_build(name):
|
||||
ssh = paramiko.SSHClient()
|
||||
ssh.set_missing_host_key_policy(paramiko.AutoAddPolicy())
|
||||
|
||||
key_file = open(os.path.join(os.path.dirname(__file__), "id_rsa_public"))
|
||||
key = paramiko.RSAKey.from_private_key(key_file)
|
||||
|
||||
print("SSH to phone {}".format(name))
|
||||
|
||||
# Try connecting for one minute
|
||||
t_start = time.time()
|
||||
while True:
|
||||
try:
|
||||
ssh.connect(hostname=name, port=8022, pkey=key, timeout=10)
|
||||
except (paramiko.ssh_exception.SSHException, socket.timeout, paramiko.ssh_exception.NoValidConnectionsError):
|
||||
print("Connection failed")
|
||||
if time.time() - t_start > 60:
|
||||
raise
|
||||
else:
|
||||
break
|
||||
time.sleep(1)
|
||||
|
||||
conn = ssh.invoke_shell()
|
||||
branch = os.environ['GIT_BRANCH']
|
||||
commit = os.environ.get('GIT_COMMIT', branch)
|
||||
|
||||
conn.send('uname -a\n')
|
||||
|
||||
conn.send('cd /data/openpilot_source\n')
|
||||
conn.send("git reset --hard\n")
|
||||
conn.send("git fetch origin\n")
|
||||
conn.send("git checkout %s\n" % commit)
|
||||
conn.send("git clean -xdf\n")
|
||||
conn.send("git submodule update --init\n")
|
||||
conn.send("git submodule foreach --recursive git reset --hard\n")
|
||||
conn.send("git submodule foreach --recursive git clean -xdf\n")
|
||||
conn.send("echo \"git took $SECONDS seconds\"\n")
|
||||
|
||||
push = "PUSH=one-master" if branch == "master" else ""
|
||||
|
||||
conn.send("%s /data/openpilot_source/release/go.sh\n" % push)
|
||||
conn.send('echo "RESULT:" $?\n')
|
||||
conn.send("exit\n")
|
||||
return conn
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
eon_name = os.environ.get('eon_name', None)
|
||||
|
||||
conn = start_build(eon_name)
|
||||
|
||||
dat = b""
|
||||
|
||||
while True:
|
||||
recvd = conn.recv(4096)
|
||||
if len(recvd) == 0:
|
||||
break
|
||||
|
||||
dat += recvd
|
||||
sys.stdout.buffer.write(recvd)
|
||||
sys.stdout.flush()
|
||||
|
||||
returns = re.findall(rb'^RESULT: (\d+)', dat[-1024:], flags=re.MULTILINE)
|
||||
sys.exit(int(returns[0]))
|
|
@ -1,3 +1,4 @@
|
|||
# flake8: noqa
|
||||
import os
|
||||
os.environ['FAKEUPLOAD'] = "1"
|
||||
|
||||
|
@ -72,26 +73,6 @@ def with_apks():
|
|||
return wrap
|
||||
return wrapper
|
||||
|
||||
#@phone_only
|
||||
#@with_processes(['controlsd', 'radard'])
|
||||
#def test_controls():
|
||||
# from selfdrive.test.longitudinal_maneuvers.plant import Plant
|
||||
#
|
||||
# # start the fake car for 2 seconds
|
||||
# plant = Plant(100)
|
||||
# for i in range(200):
|
||||
# if plant.rk.frame >= 20 and plant.rk.frame <= 25:
|
||||
# cruise_buttons = CruiseButtons.RES_ACCEL
|
||||
# # rolling forward
|
||||
# assert plant.speed > 0
|
||||
# else:
|
||||
# cruise_buttons = 0
|
||||
# plant.step(cruise_buttons = cruise_buttons)
|
||||
# plant.close()
|
||||
#
|
||||
# # assert that we stopped
|
||||
# assert plant.speed == 0.0
|
||||
|
||||
@phone_only
|
||||
@with_processes(['loggerd', 'logmessaged', 'tombstoned', 'proclogd', 'logcatd'])
|
||||
def test_logging():
|
||||
|
@ -124,118 +105,118 @@ def test_uploader():
|
|||
print("UPLOADER")
|
||||
time.sleep(10.0)
|
||||
|
||||
@phone_only
|
||||
def test_athena():
|
||||
print("ATHENA")
|
||||
start_daemon_process("manage_athenad")
|
||||
params = Params()
|
||||
manage_athenad_pid = params.get("AthenadPid")
|
||||
assert manage_athenad_pid is not None
|
||||
try:
|
||||
os.kill(int(manage_athenad_pid), 0)
|
||||
# process is running
|
||||
except OSError:
|
||||
assert False, "manage_athenad is dead"
|
||||
# @phone_only
|
||||
# def test_athena():
|
||||
# print("ATHENA")
|
||||
# start_daemon_process("manage_athenad")
|
||||
# params = Params()
|
||||
# manage_athenad_pid = params.get("AthenadPid")
|
||||
# assert manage_athenad_pid is not None
|
||||
# try:
|
||||
# os.kill(int(manage_athenad_pid), 0)
|
||||
# # process is running
|
||||
# except OSError:
|
||||
# assert False, "manage_athenad is dead"
|
||||
|
||||
def expect_athena_starts(timeout=30):
|
||||
now = time.time()
|
||||
athenad_pid = None
|
||||
while athenad_pid is None:
|
||||
try:
|
||||
athenad_pid = subprocess.check_output(["pgrep", "-P", manage_athenad_pid], encoding="utf-8").strip()
|
||||
return athenad_pid
|
||||
except subprocess.CalledProcessError:
|
||||
if time.time() - now > timeout:
|
||||
assert False, f"Athena did not start within {timeout} seconds"
|
||||
time.sleep(0.5)
|
||||
# def expect_athena_starts(timeout=30):
|
||||
# now = time.time()
|
||||
# athenad_pid = None
|
||||
# while athenad_pid is None:
|
||||
# try:
|
||||
# athenad_pid = subprocess.check_output(["pgrep", "-P", manage_athenad_pid], encoding="utf-8").strip()
|
||||
# return athenad_pid
|
||||
# except subprocess.CalledProcessError:
|
||||
# if time.time() - now > timeout:
|
||||
# assert False, f"Athena did not start within {timeout} seconds"
|
||||
# time.sleep(0.5)
|
||||
|
||||
def athena_post(payload, max_retries=5, wait=5):
|
||||
tries = 0
|
||||
while 1:
|
||||
try:
|
||||
resp = requests.post(
|
||||
"https://athena.comma.ai/" + params.get("DongleId", encoding="utf-8"),
|
||||
headers={
|
||||
"Authorization": "JWT " + os.getenv("COMMA_JWT"),
|
||||
"Content-Type": "application/json"
|
||||
},
|
||||
data=json.dumps(payload),
|
||||
timeout=30
|
||||
)
|
||||
resp_json = resp.json()
|
||||
if resp_json.get('error'):
|
||||
raise Exception(resp_json['error'])
|
||||
return resp_json
|
||||
except Exception as e:
|
||||
time.sleep(wait)
|
||||
tries += 1
|
||||
if tries == max_retries:
|
||||
raise
|
||||
else:
|
||||
print(f'athena_post failed {e}. retrying...')
|
||||
# def athena_post(payload, max_retries=5, wait=5):
|
||||
# tries = 0
|
||||
# while 1:
|
||||
# try:
|
||||
# resp = requests.post(
|
||||
# "https://athena.comma.ai/" + params.get("DongleId", encoding="utf-8"),
|
||||
# headers={
|
||||
# "Authorization": "JWT " + os.getenv("COMMA_JWT"),
|
||||
# "Content-Type": "application/json"
|
||||
# },
|
||||
# data=json.dumps(payload),
|
||||
# timeout=30
|
||||
# )
|
||||
# resp_json = resp.json()
|
||||
# if resp_json.get('error'):
|
||||
# raise Exception(resp_json['error'])
|
||||
# return resp_json
|
||||
# except Exception as e:
|
||||
# time.sleep(wait)
|
||||
# tries += 1
|
||||
# if tries == max_retries:
|
||||
# raise
|
||||
# else:
|
||||
# print(f'athena_post failed {e}. retrying...')
|
||||
|
||||
def expect_athena_registers():
|
||||
resp = athena_post({
|
||||
"method": "echo",
|
||||
"params": ["hello"],
|
||||
"id": 0,
|
||||
"jsonrpc": "2.0"
|
||||
}, max_retries=12, wait=5)
|
||||
assert resp.get('result') == "hello", f'Athena failed to register ({resp})'
|
||||
# def expect_athena_registers():
|
||||
# resp = athena_post({
|
||||
# "method": "echo",
|
||||
# "params": ["hello"],
|
||||
# "id": 0,
|
||||
# "jsonrpc": "2.0"
|
||||
# }, max_retries=12, wait=5)
|
||||
# assert resp.get('result') == "hello", f'Athena failed to register ({resp})'
|
||||
|
||||
try:
|
||||
athenad_pid = expect_athena_starts()
|
||||
# kill athenad and ensure it is restarted (check_output will throw if it is not)
|
||||
os.kill(int(athenad_pid), signal.SIGINT)
|
||||
expect_athena_starts()
|
||||
# try:
|
||||
# athenad_pid = expect_athena_starts()
|
||||
# # kill athenad and ensure it is restarted (check_output will throw if it is not)
|
||||
# os.kill(int(athenad_pid), signal.SIGINT)
|
||||
# expect_athena_starts()
|
||||
|
||||
if not os.getenv('COMMA_JWT'):
|
||||
print('WARNING: COMMA_JWT env not set, will not test requests to athena.comma.ai')
|
||||
return
|
||||
# if not os.getenv('COMMA_JWT'):
|
||||
# print('WARNING: COMMA_JWT env not set, will not test requests to athena.comma.ai')
|
||||
# return
|
||||
|
||||
expect_athena_registers()
|
||||
# expect_athena_registers()
|
||||
|
||||
print("ATHENA: getSimInfo")
|
||||
resp = athena_post({
|
||||
"method": "getSimInfo",
|
||||
"id": 0,
|
||||
"jsonrpc": "2.0"
|
||||
})
|
||||
assert resp.get('result'), resp
|
||||
assert 'sim_id' in resp['result'], resp['result']
|
||||
# print("ATHENA: getSimInfo")
|
||||
# resp = athena_post({
|
||||
# "method": "getSimInfo",
|
||||
# "id": 0,
|
||||
# "jsonrpc": "2.0"
|
||||
# })
|
||||
# assert resp.get('result'), resp
|
||||
# assert 'sim_id' in resp['result'], resp['result']
|
||||
|
||||
print("ATHENA: takeSnapshot")
|
||||
resp = athena_post({
|
||||
"method": "takeSnapshot",
|
||||
"id": 0,
|
||||
"jsonrpc": "2.0"
|
||||
})
|
||||
assert resp.get('result'), resp
|
||||
assert resp['result']['jpegBack'], resp['result']
|
||||
# print("ATHENA: takeSnapshot")
|
||||
# resp = athena_post({
|
||||
# "method": "takeSnapshot",
|
||||
# "id": 0,
|
||||
# "jsonrpc": "2.0"
|
||||
# })
|
||||
# assert resp.get('result'), resp
|
||||
# assert resp['result']['jpegBack'], resp['result']
|
||||
|
||||
@with_processes(["thermald"])
|
||||
def test_athena_thermal():
|
||||
print("ATHENA: getMessage(thermal)")
|
||||
resp = athena_post({
|
||||
"method": "getMessage",
|
||||
"params": {"service": "thermal", "timeout": 5000},
|
||||
"id": 0,
|
||||
"jsonrpc": "2.0"
|
||||
})
|
||||
assert resp.get('result'), resp
|
||||
assert resp['result']['thermal'], resp['result']
|
||||
test_athena_thermal()
|
||||
finally:
|
||||
try:
|
||||
athenad_pid = subprocess.check_output(["pgrep", "-P", manage_athenad_pid], encoding="utf-8").strip()
|
||||
except subprocess.CalledProcessError:
|
||||
athenad_pid = None
|
||||
# @with_processes(["thermald"])
|
||||
# def test_athena_thermal():
|
||||
# print("ATHENA: getMessage(thermal)")
|
||||
# resp = athena_post({
|
||||
# "method": "getMessage",
|
||||
# "params": {"service": "thermal", "timeout": 5000},
|
||||
# "id": 0,
|
||||
# "jsonrpc": "2.0"
|
||||
# })
|
||||
# assert resp.get('result'), resp
|
||||
# assert resp['result']['thermal'], resp['result']
|
||||
# test_athena_thermal()
|
||||
# finally:
|
||||
# try:
|
||||
# athenad_pid = subprocess.check_output(["pgrep", "-P", manage_athenad_pid], encoding="utf-8").strip()
|
||||
# except subprocess.CalledProcessError:
|
||||
# athenad_pid = None
|
||||
|
||||
try:
|
||||
os.kill(int(manage_athenad_pid), signal.SIGINT)
|
||||
os.kill(int(athenad_pid), signal.SIGINT)
|
||||
except (OSError, TypeError):
|
||||
pass
|
||||
# try:
|
||||
# os.kill(int(manage_athenad_pid), signal.SIGINT)
|
||||
# os.kill(int(athenad_pid), signal.SIGINT)
|
||||
# except (OSError, TypeError):
|
||||
# pass
|
||||
|
||||
# TODO: re-enable when jenkins test has /data/pythonpath -> /data/openpilot
|
||||
# @phone_only
|
||||
|
|
Loading…
Reference in New Issue