pipeline {
agent { label 'maven' }
options {
timestamps()
ansiColor('xterm')
}
stages {
stage('Prepare Metadata') {
steps {
script {
wrap([$class: 'BuildUser']) {
env.RUN_TRIGGER_USER = env.BUILD_USER_ID ?: env.BUILD_USER ?: 'system'
env.RUN_TRIGGER_NAME = env.BUILD_USER ?: env.BUILD_USER_ID ?: 'system'
}
currentBuild.displayName = "#${env.BUILD_NUMBER} ${params.BROWSER}/${params.HEADLESS}"
currentBuild.description =
"by=${env.RUN_TRIGGER_NAME}; repo=${params.QA_REPO_URL}; ref=${params.QA_REPO_REF}; mode=${params.EXECUTION_MODE}"
}
}
}
stage('Run Selenium Tests In Docker') {
steps {
sh '''
set -eux
git config --global --add safe.directory '*' || true
rm -rf ./sources
git clone "${QA_REPO_URL}" ./sources
git -C ./sources checkout "${QA_REPO_REF}"
GIT_SHA="$(git -C ./sources rev-parse --short HEAD)"
REMOTE_FACTORY="./sources/src/main/java/ru/kovbasa/driver/RemoteDriverFactory.java"
if [ -f "${REMOTE_FACTORY}" ] && ! grep -q "selenoid.video.enabled" "${REMOTE_FACTORY}"; then
sed -i 's/"enableVideo", false/"enableVideo", Boolean.parseBoolean(System.getProperty("selenoid.video.enabled", "false"))/' "${REMOTE_FACTORY}"
fi
if ! grep -q "allure-junit5" ./sources/pom.xml; then
awk -v ver="${ALLURE_ADAPTER_VERSION}" '
index($0, "") && !done {
print " "
print " io.qameta.allure"
print " allure-junit5"
print " " ver ""
print " test"
print " "
done=1
}
{ print }
' ./sources/pom.xml > ./sources/pom.xml.tmp
mv ./sources/pom.xml.tmp ./sources/pom.xml
fi
rm -rf ./artifacts
mkdir -p ./artifacts
{
echo "job=${JOB_NAME}"
echo "build=${BUILD_NUMBER}"
echo "trigger_user=${RUN_TRIGGER_USER}"
echo "trigger_name=${RUN_TRIGGER_NAME}"
echo "qa_repo_url=${QA_REPO_URL}"
echo "qa_repo_ref=${QA_REPO_REF}"
echo "qa_repo_sha=${GIT_SHA}"
echo "browser=${BROWSER}"
echo "headless=${HEADLESS}"
echo "execution_mode=${EXECUTION_MODE}"
echo "selenoid_url=${SELENOID_URL}"
echo "base_url=${BASE_URL}"
echo "timestamp_utc=$(date -u +%Y-%m-%dT%H:%M:%SZ)"
} > ./artifacts/run-info.txt
EXTRA_ARGS=""
if [ "${BROWSER}" = "chrome" ]; then
EXTRA_ARGS="-Dchrome.binary=/usr/bin/chromium"
fi
VIDEO_ARGS=""
SELENOID_BASE=""
SELENOID_VIDEO_BEFORE="./artifacts/selenoid-videos-before.txt"
SELENOID_VIDEO_AFTER="./artifacts/selenoid-videos-after.txt"
SELENOID_VIDEO_NEW="./artifacts/selenoid-videos-new.txt"
: > "${SELENOID_VIDEO_BEFORE}"
: > "${SELENOID_VIDEO_AFTER}"
: > "${SELENOID_VIDEO_NEW}"
if [ "${EXECUTION_MODE}" = "selenoid" ]; then
case "${BROWSER}" in
chrome) docker pull selenoid/vnc:chrome_128.0 || true ;;
firefox) docker pull selenoid/vnc:firefox_125.0 || true ;;
esac
docker pull selenoid/video-recorder:latest-release || docker pull selenoid/video-recorder:latest || true
VIDEO_ARGS="-Dselenoid.video.enabled=true"
if [ -n "${SELENOID_URL}" ]; then
SELENOID_BASE="${SELENOID_URL%/wd/hub}"
(curl -fsS "${SELENOID_BASE}/video/" || true) \
| tr '"' '\n' \
| grep -E '\\.mp4$' \
| sort -u > "${SELENOID_VIDEO_BEFORE}" || true
fi
fi
CID="$(docker create \
--add-host=host.docker.internal:host-gateway \
localhost:5005/otus/test-selenium:1.0.0 \
bash -lc "set -e; cd /workspace; mvn -Dallure.version=${ALLURE_ADAPTER_VERSION} -Dexecution.mode=${EXECUTION_MODE} -Dbrowser=${BROWSER} -Dbrowser.version= -Dselenoid.url=${SELENOID_URL} -Dbase.url=${BASE_URL} -Dselenide.headless=${HEADLESS} -Dallure.results.directory=target/allure-results ${VIDEO_ARGS} ${EXTRA_ARGS} test")"
cleanup_container() {
docker rm -f "${CID}" >/dev/null 2>&1 || true
}
trap cleanup_container EXIT INT TERM
tar -C ./sources -cf - . | docker cp - "${CID}:/workspace"
set +e
docker start -a "${CID}"
TEST_RC=$?
docker cp "${CID}:/workspace/target" "./artifacts" || true
mkdir -p ./artifacts/target/allure-results
if [ "${EXECUTION_MODE}" = "selenoid" ] && [ -n "${SELENOID_URL}" ]; then
(curl -fsS "${SELENOID_BASE}/video/" || true) \
| tr '"' '\n' \
| grep -E '\\.mp4$' \
| sort -u > "${SELENOID_VIDEO_AFTER}" || true
comm -13 "${SELENOID_VIDEO_BEFORE}" "${SELENOID_VIDEO_AFTER}" > "${SELENOID_VIDEO_NEW}" || true
if [ ! -s "${SELENOID_VIDEO_NEW}" ]; then
cp "${SELENOID_VIDEO_AFTER}" "${SELENOID_VIDEO_NEW}" || true
fi
while IFS= read -r video_file; do
[ -n "${video_file}" ] || continue
downloaded=0
for i in $(seq 1 20); do
if curl -fsS "${SELENOID_BASE}/video/${video_file}" -o "./artifacts/target/${video_file}"; then
downloaded=1
break
fi
sleep 2
done
if [ "${downloaded}" -ne 1 ]; then
echo "Failed to fetch Selenoid video: ${video_file}"
fi
done < "${SELENOID_VIDEO_NEW}"
fi
ATTACHMENT_COUNT=0
ATTACHMENTS_MANIFEST="./artifacts/target/allure-results/.external-attachments.txt"
: > "${ATTACHMENTS_MANIFEST}"
ATTACH_FILES="$(find ./artifacts -type f \\( -iname '*.png' -o -iname '*.jpg' -o -iname '*.jpeg' -o -iname '*.mp4' -o -iname '*.webm' \\) 2>/dev/null || true)"
if [ -n "${ATTACH_FILES}" ]; then
while IFS= read -r attachment_file; do
if [ -z "${attachment_file}" ] || [ ! -f "${attachment_file}" ]; then
continue
fi
ATTACHMENT_COUNT=$((ATTACHMENT_COUNT + 1))
ext="$(echo "${attachment_file}" | awk -F. '{print tolower($NF)}')"
mime="application/octet-stream"
case "${ext}" in
png) mime="image/png" ;;
jpg|jpeg) mime="image/jpeg" ;;
mp4) mime="video/mp4" ;;
webm) mime="video/webm" ;;
esac
source_name="external-attachment-${ATTACHMENT_COUNT}.${ext}"
cp "${attachment_file}" "./artifacts/target/allure-results/${source_name}" || continue
safe_name="$(basename "${attachment_file}" | sed 's/"/\\"/g')"
printf '%s|%s|%s\n' "${safe_name}" "${mime}" "${source_name}" >> "${ATTACHMENTS_MANIFEST}"
done < "./artifacts/target/allure-results/${EXTRA_UUID}-result.json" < ./artifacts/target/allure-results/environment.properties
cat > ./artifacts/target/allure-results/executor.json < ./artifacts/target/allure-results/categories.json <<'EOF'
[
{
"name": "Infrastructure issues",
"matchedStatuses": ["broken"],
"messageRegex": ".*(Connection refused|No route to host|timed out|timeout).*"
},
{
"name": "UI locator/assertion issues",
"matchedStatuses": ["failed", "broken"],
"messageRegex": ".*(Element not found|NoSuchElement|AssertionError).*"
}
]
EOF
trap - EXIT INT TERM
docker rm -f "${CID}" || true
exit "${TEST_RC}"
'''
}
}
}
post {
always {
script {
try {
allure commandline: 'allure', includeProperties: false, jdk: '', results: [[path: 'artifacts/target/allure-results']]
} catch (Exception ex) {
echo "Allure publisher unavailable: ${ex.message}"
}
}
junit allowEmptyResults: true, testResults: 'artifacts/target/surefire-reports/*.xml'
archiveArtifacts allowEmptyArchive: true, artifacts: 'artifacts/run-info.txt,artifacts/target/**'
}
}
}