diff --git a/ghaf-hw-test.groovy b/ghaf-hw-test.groovy index 3123d49..16a90e8 100644 --- a/ghaf-hw-test.groovy +++ b/ghaf-hw-test.groovy @@ -208,7 +208,6 @@ pipeline { unmount_cmd = "/run/wrappers/bin/sudo usbsdmux ${muxport} dut" } else { serial = get_test_conf_property(CONF_FILE_PATH, env.DEVICE_NAME, 'usbhub_serial') - dgrep = 'PSSD' mount_cmd = "/run/wrappers/bin/sudo AcronameHubCLI -u 0 -s ${serial}; sleep 10" unmount_cmd = "/run/wrappers/bin/sudo AcronameHubCLI -u 1 -s ${serial}" } @@ -216,7 +215,7 @@ pipeline { // Mount the target disk sh "${mount_cmd}" // Read the device name - dev = run_cmd("lsblk -o model,name | grep '${dgrep}' | rev | cut -d ' ' -f 1 | rev | grep .") + dev = get_test_conf_property(CONF_FILE_PATH, env.DEVICE_NAME, 'ext_drive_by-id') println "Using device '$dev'" // Wipe possible ZFS leftovers, more details here: // https://github.com/tiiuae/ghaf/blob/454b18bc/packages/installer/ghaf-installer.sh#L75 @@ -226,18 +225,18 @@ pipeline { SECTOR = 512 MIB_TO_SECTORS = 20480 // Disk size in 512-byte sectors - SECTORS = sh(script: "/run/wrappers/bin/sudo blockdev --getsz /dev/${dev}", returnStdout: true).trim() + SECTORS = sh(script: "/run/wrappers/bin/sudo blockdev --getsz /dev/disk/by-id/${dev}", returnStdout: true).trim() // Unmount possible mounted filesystems - sh "sync; /run/wrappers/bin/sudo umount -q /dev/${dev}* || true" + sh "sync; /run/wrappers/bin/sudo umount -q /dev/disk/by-id/${dev}* || true" // Wipe first 10MiB of disk - sh "/run/wrappers/bin/sudo dd if=/dev/zero of=/dev/${dev} bs=${SECTOR} count=${MIB_TO_SECTORS} conv=fsync status=none" + sh "/run/wrappers/bin/sudo dd if=/dev/zero of=/dev/disk/by-id/${dev} bs=${SECTOR} count=${MIB_TO_SECTORS} conv=fsync status=none" // Wipe last 10MiB of disk - sh "/run/wrappers/bin/sudo dd if=/dev/zero of=/dev/${dev} bs=${SECTOR} count=${MIB_TO_SECTORS} seek=\$(( ${SECTORS} - ${MIB_TO_SECTORS} )) conv=fsync status=none" + sh "/run/wrappers/bin/sudo dd if=/dev/zero of=/dev/disk/by-id/${dev} bs=${SECTOR} count=${MIB_TO_SECTORS} seek=\$(( ${SECTORS} - ${MIB_TO_SECTORS} )) conv=fsync status=none" } // Write the image img_relpath = run_cmd("find ${TMP_IMG_DIR} -type f -print -quit | grep .") println "Using image '$img_relpath'" - sh "/run/wrappers/bin/sudo dd if=${img_relpath} of=/dev/${dev} bs=1M status=progress conv=fsync" + sh "/run/wrappers/bin/sudo dd if=${img_relpath} of=/dev/disk/by-id/${dev} bs=1M status=progress conv=fsync" // Unmount sh "${unmount_cmd}" } diff --git a/ghaf-nightly-pipeline.groovy b/ghaf-nightly-pipeline.groovy index 65e10df..23d02b3 100644 --- a/ghaf-nightly-pipeline.groovy +++ b/ghaf-nightly-pipeline.groovy @@ -66,23 +66,23 @@ def targets = [ ], [ system: "aarch64-linux", target: "nxp-imx8mp-evk-debug", archive: true, scs: true - ], + ], ] def hydrajobs_targets = [ // nvidia orin with bpmp enabled - [ system: "aarch64-linux",target: "nvidia-jetson-orin-agx-debug-bpmp", + [ system: "aarch64-linux",target: "nvidia-jetson-orin-agx-debug-bpmp", archive: true - ], - [ system: "aarch64-linux",target: "nvidia-jetson-orin-nx-debug-bpmp", + ], + [ system: "aarch64-linux",target: "nvidia-jetson-orin-nx-debug-bpmp", archive: true - ], - [ system: "x86_64-linux", target: "nvidia-jetson-orin-agx-debug-bpmp-from-x86_64", + ], + [ system: "x86_64-linux", target: "nvidia-jetson-orin-agx-debug-bpmp-from-x86_64", archive: true - ], - [ system: "x86_64-linux", target: "nvidia-jetson-orin-nx-debug-bpmp-from-x86_64", + ], + [ system: "x86_64-linux", target: "nvidia-jetson-orin-nx-debug-bpmp-from-x86_64", archive: true - ], + ], ] //////////////////////////////////////////////////////////////////////////////// diff --git a/ghaf-parallel-hw-test.groovy b/ghaf-parallel-hw-test.groovy deleted file mode 100644 index 16a90e8..0000000 --- a/ghaf-parallel-hw-test.groovy +++ /dev/null @@ -1,302 +0,0 @@ -#!/usr/bin/env groovy - -// SPDX-FileCopyrightText: 2022-2024 TII (SSRC) and the Ghaf contributors -// SPDX-License-Identifier: Apache-2.0 - -//////////////////////////////////////////////////////////////////////////////// - -def REPO_URL = 'https://github.com/tiiuae/ci-test-automation/' -def DEF_LABEL = 'testagent' -def TMP_IMG_DIR = 'image' -def TMP_SIG_DIR = 'signature' -def CONF_FILE_PATH = '/etc/jenkins/test_config.json' - -//////////////////////////////////////////////////////////////////////////////// - -def run_cmd(String cmd) { - // Run cmd returning stdout - return sh(script: cmd, returnStdout:true).trim() -} - -def get_test_conf_property(String file_path, String device, String property) { - // Get the requested device property data from test_config.json file - def device_data = readJSON file: file_path - property_data = "${device_data['addresses'][device][property]}" - println "Got device '${device}' property '${property}' value: '${property_data}'" - return property_data -} - -def ghaf_robot_test(String testname='boot') { - if (!env.DEVICE_TAG) { - sh "echo 'DEVICE_TAG not set'; exit 1" - } - if (!env.DEVICE_NAME) { - sh "echo 'DEVICE_NAME not set'; exit 1" - } - if (testname == 'turnoff') { - env.INCLUDE_TEST_TAGS = "${testname}" - } else { - env.INCLUDE_TEST_TAGS = "${testname}AND${env.DEVICE_TAG}" - } - // TODO: do we really need credentials to access the target devices? - // Target devices are connected to the testagent, which itself is - // only available over a private network. What is the risk - // we are protecting against by having additional authentication - // for the test devices? - // The current configuration requires additional manual configuration - // on the jenkins UI to add the following secrets: - withCredentials([ - string(credentialsId: 'testagent-dut-pass', variable: 'DUT_PASS'), - string(credentialsId: 'testagent-plug-pass', variable: 'PLUG_PASS'), - string(credentialsId: 'testagent-switch-token', variable: 'SW_TOKEN'), - string(credentialsId: 'testagent-switch-secret', variable: 'SW_SECRET'), - string(credentialsId: 'testagent-wifi-ssid', variable: 'WIFI_SSID'), - string(credentialsId: 'testagent-wifi-password', variable: 'WIFI_PSWD'), - ]) { - dir("Robot-Framework/test-suites") { - sh 'rm -f *.png output.xml report.html log.html' - // On failure, continue the pipeline execution - try { - // Pass the secrets to the shell as environment variables, as we - // don't want Groovy to interpolate them. Similary, we pass - // other variables as environment variables to shell. - // Ref: https://www.jenkins.io/doc/book/pipeline/jenkinsfile/#string-interpolation - sh ''' - nix run .#ghaf-robot -- \ - -v DEVICE:$DEVICE_NAME \ - -v DEVICE_TYPE:$DEVICE_TAG \ - -v LOGIN:ghaf \ - -v PASSWORD:$DUT_PASS \ - -v PLUG_USERNAME:ghaftester@gmail.com \ - -v PLUG_PASSWORD:$PLUG_PASS \ - -v SWITCH_TOKEN:$SW_TOKEN \ - -v SWITCH_SECRET:$SW_SECRET \ - -v BUILD_ID:${BUILD_NUMBER} \ - -v TEST_WIFI_SSID:${WIFI_SSID} \ - -v TEST_WIFI_PSWD:${WIFI_PSWD} \ - -i $INCLUDE_TEST_TAGS . - ''' - if (testname == 'boot') { - // Set an environment variable to indicate boot test passed - env.BOOT_PASSED = 'true' - } - } catch (Exception e) { - currentBuild.result = "FAILURE" - unstable("FAILED '${testname}': ${e.toString()}") - } finally { - // Move the test output (if any) to a subdirectory - sh """ - rm -fr $testname; mkdir -p $testname - mv -f *.png output.xml report.html log.html $testname/ || true - """ - } - } - } -} - -//////////////////////////////////////////////////////////////////////////////// - -pipeline { - agent { label "${params.getOrDefault('LABEL', DEF_LABEL)}" } - options { timestamps () } - stages { - stage('Checkout') { - steps { - checkout scmGit( - branches: [[name: 'main']], - extensions: [cleanBeforeCheckout()], - userRemoteConfigs: [[url: REPO_URL]] - ) - } - } - stage('Setup') { - steps { - script { - env.TEST_CONFIG_DIR = 'Robot-Framework/config' - if(!params.getOrDefault('TARGET', null)) { - println "Missing TARGET parameter" - sh "exit 1" - } - println "Using TARGET: ${params.TARGET}" - sh """ - mkdir -p ${TEST_CONFIG_DIR} - rm -f ${TEST_CONFIG_DIR}/*.json - ln -sv ${CONF_FILE_PATH} ${TEST_CONFIG_DIR} - echo { \\\"Job\\\": \\\"${params.TARGET}\\\" } > ${TEST_CONFIG_DIR}/${BUILD_NUMBER}.json - ls -la ${TEST_CONFIG_DIR} - """ - if(!params.containsKey('DESC')) { - println "Missing DESC parameter, skip setting description" - } else { - currentBuild.description = "${params.DESC}" - } - env.TESTSET = params.getOrDefault('TESTSET', '_boot_') - println "Using TESTSET: ${env.TESTSET}" - } - } - } - stage('Image download') { - steps { - script { - if(!params.containsKey('IMG_URL')) { - println "Missing IMG_URL parameter" - sh "exit 1" - } - sh "rm -fr ${TMP_IMG_DIR}" - // Wget occasionally fails due to a failure in name lookup. Below is a - // hack to force re-try a few times before aborting. Wget options, such - // as --tries, --waitretry, --retry-connrefused, etc. do not help in case - // the failure is due to an issue in name resolution which is considered - // a fatal error. Therefore, we need to add the below retry loop. - // TODO: remove the below re-try loop when test network DNS works - // reliably. - sh """ - retry=1 - max_retry=3 - while ! wget -nv --show-progress --progress=dot:giga -P ${TMP_IMG_DIR} ${params.IMG_URL}; - do - if (( \$retry >= \$max_retry )); then - echo "wget failed after \$retry retries" - exit 1 - fi - retry=\$(( \$retry + 1 )) - sleep 5 - done - """ - img_relpath = run_cmd("find ${TMP_IMG_DIR} -type f -print -quit | grep .") - println "Downloaded image to workspace: ${img_relpath}" - // Verify signature using the tooling from: https://github.com/tiiuae/ci-yubi - sh "wget -nv -P ${TMP_SIG_DIR} ${params.IMG_URL}.sig" - sig_relpath = run_cmd("find ${TMP_SIG_DIR} -type f -print -quit | grep .") - println "Downloaded signature to workspace: ${sig_relpath}" - sh "nix run github:tiiuae/ci-yubi/bdb2dbf#verify -- --path ${img_relpath} --sigfile ${sig_relpath} --cert INT-Ghaf-Devenv-Image" - // Uncompress, keeping only the decompressed image file - if(img_relpath.endsWith("zst")) { - sh "zstd -dfv ${img_relpath} && rm ${img_relpath}" - } - sh "ls -la ${TMP_IMG_DIR}" - } - } - } - stage('Flash') { - steps { - script { - if(!params.getOrDefault('DEVICE_CONFIG_NAME', null)) { - println "Missing DEVICE_CONFIG_NAME parameter" - sh "exit 1" - } - // Determine the device name - if(params.DEVICE_CONFIG_NAME == "orin-agx") { - env.DEVICE_NAME = 'OrinAGX1' - } else if(params.DEVICE_CONFIG_NAME == "orin-nx") { - env.DEVICE_NAME = 'OrinNX1' - } else if(params.DEVICE_CONFIG_NAME == "lenovo-x1") { - env.DEVICE_NAME = 'LenovoX1-1' - } else if(params.DEVICE_CONFIG_NAME == "nuc") { - env.DEVICE_NAME = 'NUC1' - } else if(params.DEVICE_CONFIG_NAME == "riscv") { - env.DEVICE_NAME = 'Polarfire1' - } else { - println "Error: unsupported device config '${params.DEVICE_CONFIG_NAME}'" - sh "exit 1" - } - // Determine mount commands - if(params.DEVICE_CONFIG_NAME == "riscv") { - muxport = get_test_conf_property(CONF_FILE_PATH, env.DEVICE_NAME, 'usb_sd_mux_port') - dgrep = 'sdmux' - mount_cmd = "/run/wrappers/bin/sudo usbsdmux ${muxport} host; sleep 10" - unmount_cmd = "/run/wrappers/bin/sudo usbsdmux ${muxport} dut" - } else { - serial = get_test_conf_property(CONF_FILE_PATH, env.DEVICE_NAME, 'usbhub_serial') - mount_cmd = "/run/wrappers/bin/sudo AcronameHubCLI -u 0 -s ${serial}; sleep 10" - unmount_cmd = "/run/wrappers/bin/sudo AcronameHubCLI -u 1 -s ${serial}" - } - env.DEVICE_TAG = params.DEVICE_CONFIG_NAME - // Mount the target disk - sh "${mount_cmd}" - // Read the device name - dev = get_test_conf_property(CONF_FILE_PATH, env.DEVICE_NAME, 'ext_drive_by-id') - println "Using device '$dev'" - // Wipe possible ZFS leftovers, more details here: - // https://github.com/tiiuae/ghaf/blob/454b18bc/packages/installer/ghaf-installer.sh#L75 - // TODO: use ghaf flashing scripts or installers? - if(params.DEVICE_CONFIG_NAME == "lenovo-x1") { - echo "Wiping filesystem..." - SECTOR = 512 - MIB_TO_SECTORS = 20480 - // Disk size in 512-byte sectors - SECTORS = sh(script: "/run/wrappers/bin/sudo blockdev --getsz /dev/disk/by-id/${dev}", returnStdout: true).trim() - // Unmount possible mounted filesystems - sh "sync; /run/wrappers/bin/sudo umount -q /dev/disk/by-id/${dev}* || true" - // Wipe first 10MiB of disk - sh "/run/wrappers/bin/sudo dd if=/dev/zero of=/dev/disk/by-id/${dev} bs=${SECTOR} count=${MIB_TO_SECTORS} conv=fsync status=none" - // Wipe last 10MiB of disk - sh "/run/wrappers/bin/sudo dd if=/dev/zero of=/dev/disk/by-id/${dev} bs=${SECTOR} count=${MIB_TO_SECTORS} seek=\$(( ${SECTORS} - ${MIB_TO_SECTORS} )) conv=fsync status=none" - } - // Write the image - img_relpath = run_cmd("find ${TMP_IMG_DIR} -type f -print -quit | grep .") - println "Using image '$img_relpath'" - sh "/run/wrappers/bin/sudo dd if=${img_relpath} of=/dev/disk/by-id/${dev} bs=1M status=progress conv=fsync" - // Unmount - sh "${unmount_cmd}" - } - } - } - stage('Boot test') { - when { expression { env.TESTSET.contains('_boot_')} } - steps { - script { - env.BOOT_PASSED = 'false' - ghaf_robot_test('boot') - println "Boot test passed: ${env.BOOT_PASSED}" - } - } - } - stage('Bat test') { - when { expression { env.BOOT_PASSED == 'true' && env.TESTSET.contains('_bat_')} } - steps { - script { - ghaf_robot_test('bat') - } - } - } - stage('Perf test') { - when { expression { env.BOOT_PASSED == 'true' && env.TESTSET.contains('_perf_')} } - steps { - script { - ghaf_robot_test('performance') - } - } - } - stage('Turn off') { - steps { - script { - ghaf_robot_test('turnoff') - } - } - } - } - post { - always { - // Archive Robot-Framework results as artifacts - archiveArtifacts allowEmptyArchive: true, artifacts: 'Robot-Framework/test-suites/**/*.html, Robot-Framework/test-suites/**/*.xml, Robot-Framework/test-suites/**/*.png' - // Publish all results under Robot-Framework/test-suites subfolders - step( - [$class: 'RobotPublisher', - archiveDirName: 'robot-plugin', - outputPath: 'Robot-Framework/test-suites', - outputFileName: '**/output.xml', - otherFiles: '**/*.png', - disableArchiveOutput: false, - reportFileName: '**/report.html', - logFileName: '**/log.html', - passThreshold: 0, - unstableThreshold: 0, - onlyCritical: true, - ] - ) - } - } -} - -//////////////////////////////////////////////////////////////////////////////// diff --git a/utils.groovy b/utils.groovy index ab10f6d..9bd2efa 100644 --- a/utils.groovy +++ b/utils.groovy @@ -187,67 +187,6 @@ def sign_file(String path, String sigfile, String cert="INT-Ghaf-Devenv-Common") } def ghaf_hw_test(String flakeref, String device_config, String testset='_boot_') { - testagent_nodes = nodesByLabel(label: "$device_config", offline: false) - if (!testagent_nodes) { - println "Warning: Skipping HW test '$flakeref', no test agents online" - unstable("No test agents online") - return - } - if (!env.ARTIFACTS_REMOTE_PATH) { - println "Warning: skipping HW test '$flakeref', ARTIFACTS_REMOTE_PATH not set" - return - } - if (!env.JENKINS_URL) { - println "Warning: skipping HW test '$flakeref', JENKINS_URL not set" - return - } - // Compose the image URL; testagent will need this URL to download the image - imgdir = find_img_relpath(flakeref, 'archive') - remote_path = "artifacts/${env.ARTIFACTS_REMOTE_PATH}" - img_url = "${env.JENKINS_URL}/${remote_path}/${imgdir}" - build_url = "${env.JENKINS_URL}/job/${env.JOB_NAME}/${env.BUILD_ID}" - build_href = "${env.JOB_NAME}#${env.BUILD_ID}" - flakeref_trimmed = "${flakeref_trim(flakeref)}" - // 'short' flakeref: everything after the last occurence of '.' (if any) - flakeref_short = flakeref_trimmed.replaceAll(/.*\.+/,"") - description = "Triggered by ${build_href}
(${flakeref_short})" - // Trigger a build in 'ghaf-hw-test' pipeline. - // 'build' step is documented in https://plugins.jenkins.io/pipeline-build-step/ - job = build( - job: "ghaf-hw-test", - propagate: false, - parameters: [ - string(name: "LABEL", value: "$device_config"), - string(name: "DEVICE_CONFIG_NAME", value: "$device_config"), - string(name: "IMG_URL", value: "$img_url"), - string(name: "DESC", value: "$description"), - string(name: "TESTSET", value: "$testset"), - string(name: "TARGET", value: "$flakeref_trimmed"), - ], - wait: true, - ) - println "ghaf-hw-test result (${device_config}:${testset}): ${job.result}" - // If the test job failed, mark the current step unstable and set - // the final build result failed, but continue the pipeline execution. - if (job.result != "SUCCESS") { - unstable("FAILED: ${device_config} ${testset}") - currentBuild.result = "FAILURE" - // Add a link to failed test job(s) on the calling pipeline - test_href = "⛔ ${flakeref_short}" - currentBuild.description = "${currentBuild.description}
${test_href}" - return - } - // Copy test results from agent to controller to 'test-results' directory - copyArtifacts( - projectName: "ghaf-hw-test", - selector: specific("${job.number}"), - target: "ghaf-hw-test/${flakeref_trimmed}/test-results", - ) - // Archive the test results - archive_artifacts("ghaf-hw-test") -} - -def ghaf_parallel_hw_test(String flakeref, String device_config, String testset='_boot_') { testagent_nodes = nodesByLabel(label: "$device_config", offline: false) if (!testagent_nodes) { println "Warning: Skipping HW test '$flakeref', no test agents online" @@ -270,10 +209,10 @@ def ghaf_parallel_hw_test(String flakeref, String device_config, String testset= def build_href = "${env.JOB_NAME}#${env.BUILD_ID}" def flakeref_trimmed = "${flakeref_trim(flakeref)}" def description = "Triggered by ${build_href}
(${flakeref_trimmed})" - // Trigger a build in 'ghaf-parallel-hw-test' pipeline. + // Trigger a build in 'ghaf-hw-test' pipeline. // 'build' step is documented in https://plugins.jenkins.io/pipeline-build-step/ def job = build( - job: "ghaf-parallel-hw-test", + job: "ghaf-hw-test", propagate: false, parameters: [ string(name: "LABEL", value: "$device_config"), @@ -285,7 +224,7 @@ def ghaf_parallel_hw_test(String flakeref, String device_config, String testset= ], wait: true, ) - println "ghaf-parallel-hw-test result (${device_config}:${testset}): ${job.result}" + println "ghaf-hw-test result (${device_config}:${testset}): ${job.result}" // If the test job failed, mark the current step unstable and set // the final build result failed, but continue the pipeline execution. if (job.result != "SUCCESS") { @@ -298,12 +237,12 @@ def ghaf_parallel_hw_test(String flakeref, String device_config, String testset= } // Copy test results from agent to controller to 'test-results' directory copyArtifacts( - projectName: "ghaf-parallel-hw-test", + projectName: "ghaf-hw-test", selector: specific("${job.number}"), - target: "ghaf-parallel-hw-test/${flakeref_trimmed}/test-results", + target: "ghaf-hw-test/${flakeref_trimmed}/test-results", ) // Archive the test results - archive_artifacts("ghaf-parallel-hw-test", flakeref_trimmed) + archive_artifacts("ghaf-hw-test", flakeref_trimmed) } def nix_eval_jobs(List targets) { @@ -315,7 +254,7 @@ def nix_eval_jobs(List targets) { aarch64_targets = aarch64_targets ? "\"${aarch64_targets.join('" "')}\"" : "" // nix-eval-jobs is used to evaluate the targets in parallel and compute derivation paths. - // nix expression is used to create an attset on the fly which is a subset of #packages, + // nix expression is used to create an attset on the fly which is a subset of #packages, // but optimized to only include the targets we want to build sh """ nix-eval-jobs --gc-roots-dir gcroots --force-recurse --expr ' \ @@ -374,7 +313,7 @@ def create_parallel_stages(List targets, Boolean skip_hw_test=false) { target_jobs[displayName] = { stage("Build ${displayName}") { def opts = "" - if (it.archive) { + if (it.archive) { opts = "--out-link archive/${targetAttr}" } else { opts = "--no-link" @@ -419,7 +358,7 @@ def create_parallel_stages(List targets, Boolean skip_hw_test=false) { }, "job": "${env.JOB_NAME}", "jobParams": ${JsonOutput.toJson(params)}, - "buildRun": "${env.BUILD_ID}" + "buildRun": "${env.BUILD_ID}" } """ // this environment block is only valid for the scope of this stage, @@ -436,7 +375,7 @@ def create_parallel_stages(List targets, Boolean skip_hw_test=false) { def outpath = "${scsdir}/provenance.json" sh """ mkdir -p ${scsdir} - provenance ${it.drvPath} --recursive --out ${outpath} + provenance ${it.drvPath} --recursive --out ${outpath} """ sign_file(outpath, "sig/${outpath}.sig", "INT-Ghaf-Devenv-Provenance") } @@ -479,7 +418,7 @@ def create_parallel_stages(List targets, Boolean skip_hw_test=false) { if (!skip_hw_test && it.hwtest_device != null) { stage("Test ${displayName}") { script { - ghaf_parallel_hw_test(targetAttr, it.hwtest_device, '_boot_bat_perf_') + ghaf_hw_test(targetAttr, it.hwtest_device, '_boot_bat_perf_') } } }