xref: /openbmc/openbmc-build-scripts/run-unit-test-docker.sh (revision b8c7c169f7d82c0d02ce0c1e58691a8831780e2c)
1#!/bin/bash -xe
2
3# This build script is for running the Jenkins unit test builds using docker.
4#
5# This script will build a docker container which will then be used to build
6# and test the input UNIT_TEST_PKG. The docker container will be pre-populated
7# with the most used OpenBMC repositories (phosphor-dbus-interfaces, sdbusplus,
8# phosphor-logging, ...). This allows the use of docker caching
9# capabilities so the dependent repositories are only built once per update
10# to their corresponding repository. If a BRANCH parameter is input then the
11# docker container will be pre-populated with the latest code from that input
12# branch. If the branch does not exist in the repository, then master will be
13# used.
14#
15#   UNIT_TEST_PKG:   Required, repository which has been extracted and is to
16#                    be tested
17#   WORKSPACE:       Required, location of unit test scripts and repository
18#                    code to test
19#   BRANCH:          Optional, branch to build from each of the
20#                    openbmc repositories. default is master, which will be
21#                    used if input branch not provided or not found
22#   dbus_sys_config_file: Optional, with the default being
23#                         `/usr/share/dbus-1/system.conf`
24#   TEST_ONLY:       Optional, do not run analysis tools
25#   NO_FORMAT_CODE:  Optional, do not run format-code.sh
26#   EXTRA_UNIT_TEST_ARGS:  Optional, pass arguments to unit-test.py
27#   INTERACTIVE: Optional, run a bash shell instead of unit-test.py
28#   http_proxy: Optional, run the container with proxy environment
29
30# Trace bash processing. Set -e so when a step fails, we fail the build
31set -uo pipefail
32
33# Default variables
34BRANCH=${BRANCH:-"master"}
35OBMC_BUILD_SCRIPTS="openbmc-build-scripts"
36UNIT_TEST_PY_DIR="scripts"
37CONFIG_DIR="config"
38UNIT_TEST_PY="unit-test.py"
39FORMAT_CODE_SH="format-code.sh"
40SPELLINGS_TXT="openbmc-spelling.txt"
41ESLINT_CONFIG="eslint-global-config.json"
42DBUS_UNIT_TEST_PY="dbus-unit-test.py"
43TEST_ONLY="${TEST_ONLY:-}"
44DBUS_SYS_CONFIG_FILE=${dbus_sys_config_file:-"/usr/share/dbus-1/system.conf"}
45MAKEFLAGS="${MAKEFLAGS:-""}"
46DOCKER_WORKDIR="${DOCKER_WORKDIR:-$WORKSPACE}"
47NO_FORMAT_CODE="${NO_FORMAT_CODE:-}"
48INTERACTIVE="${INTERACTIVE:-}"
49http_proxy=${http_proxy:-}
50
51# Timestamp for job
52echo "Unit test build started, $(date)"
53
54# Check workspace, build scripts, and package to be unit tested exists
55if [ ! -d "${WORKSPACE}" ]; then
56    echo "Workspace(${WORKSPACE}) doesn't exist, exiting..."
57    exit 1
58fi
59if [ ! -d "${WORKSPACE}/${OBMC_BUILD_SCRIPTS}" ]; then
60    echo "Package(${OBMC_BUILD_SCRIPTS}) not found in ${WORKSPACE}, exiting..."
61    exit 1
62fi
63# shellcheck disable=SC2153 # UNIT_TEST_PKG is not misspelled.
64if [ ! -d "${WORKSPACE}/${UNIT_TEST_PKG}" ]; then
65    echo "Package(${UNIT_TEST_PKG}) not found in ${WORKSPACE}, exiting..."
66    exit 1
67fi
68
69# Copy unit test script into workspace
70cp "${WORKSPACE}"/${OBMC_BUILD_SCRIPTS}/${UNIT_TEST_PY_DIR}/${UNIT_TEST_PY} \
71"${WORKSPACE}"/${UNIT_TEST_PY}
72chmod a+x "${WORKSPACE}"/${UNIT_TEST_PY}
73
74# Copy dbus unit test script into workspace
75cp "${WORKSPACE}"/${OBMC_BUILD_SCRIPTS}/${UNIT_TEST_PY_DIR}/${DBUS_UNIT_TEST_PY} \
76"${WORKSPACE}"/${DBUS_UNIT_TEST_PY}
77chmod a+x "${WORKSPACE}"/${DBUS_UNIT_TEST_PY}
78
79# Copy format code script into workspace
80cp "${WORKSPACE}"/${OBMC_BUILD_SCRIPTS}/${UNIT_TEST_PY_DIR}/${FORMAT_CODE_SH} \
81"${WORKSPACE}"/${FORMAT_CODE_SH}
82chmod a+x "${WORKSPACE}"/${FORMAT_CODE_SH}
83
84# Copy spellings.txt file into workspace
85cp "${WORKSPACE}"/${OBMC_BUILD_SCRIPTS}/${CONFIG_DIR}/${SPELLINGS_TXT} \
86"${WORKSPACE}"/${SPELLINGS_TXT}
87
88# Copy the eslintconfig file into workspce
89cp "${WORKSPACE}"/${OBMC_BUILD_SCRIPTS}/${CONFIG_DIR}/${ESLINT_CONFIG} \
90"${WORKSPACE}"/${ESLINT_CONFIG}
91
92# Configure docker build
93cd "${WORKSPACE}"/${OBMC_BUILD_SCRIPTS}
94echo "Building docker image with build-unit-test-docker"
95# Export input env variables
96export BRANCH
97DOCKER_IMG_NAME=$(./scripts/build-unit-test-docker)
98export DOCKER_IMG_NAME
99
100# Allow the user to pass options through to unit-test.py:
101#   EXTRA_UNIT_TEST_ARGS="-r 100" ...
102EXTRA_UNIT_TEST_ARGS="${EXTRA_UNIT_TEST_ARGS:+,${EXTRA_UNIT_TEST_ARGS/ /,}}"
103
104# Unit test and parameters
105if [ "${INTERACTIVE}" ]; then
106    UNIT_TEST="/bin/bash"
107else
108    UNIT_TEST="${DOCKER_WORKDIR}/${UNIT_TEST_PY},-w,${DOCKER_WORKDIR},\
109-p,${UNIT_TEST_PKG},-b,$BRANCH,-v${TEST_ONLY:+,-t}${NO_FORMAT_CODE:+,-n}\
110${EXTRA_UNIT_TEST_ARGS}"
111fi
112
113# Run the docker unit test container with the unit test execution script
114echo "Executing docker image"
115
116PROXY_ENV=""
117# Set up proxies
118if [ -n "${http_proxy}" ]; then
119    PROXY_ENV=" \
120        --env HTTP_PROXY=${http_proxy} \
121        --env HTTPS_PROXY=${http_proxy} \
122        --env FTP_PROXY=${http_proxy} \
123        --env http_proxy=${http_proxy} \
124        --env https_proxy=${http_proxy} \
125        --env ftp_proxy=${http_proxy}"
126fi
127
128# shellcheck disable=SC2086 # ${PROXY_ENV} is meant to be splitted
129docker run --cap-add=sys_admin --rm=true \
130    --privileged=true \
131    ${PROXY_ENV} \
132    -u "$USER" \
133    -w "${DOCKER_WORKDIR}" -v "${WORKSPACE}":"${DOCKER_WORKDIR}" \
134    -e "MAKEFLAGS=${MAKEFLAGS}" \
135    -${INTERACTIVE:+i}t "${DOCKER_IMG_NAME}" \
136    "${DOCKER_WORKDIR}"/${DBUS_UNIT_TEST_PY} -u "${UNIT_TEST}" \
137    -f "${DBUS_SYS_CONFIG_FILE}"
138
139# Timestamp for build
140echo "Unit test build completed, $(date)"
141
142# Clean up copied scripts.
143rm "${WORKSPACE}"/${UNIT_TEST_PY}
144rm "${WORKSPACE}"/${DBUS_UNIT_TEST_PY}
145rm "${WORKSPACE}"/${FORMAT_CODE_SH}
146rm "${WORKSPACE}"/${ESLINT_CONFIG}
147
148