Skip to content

test(e2e): migrate device auth health to vitest #9133

test(e2e): migrate device auth health to vitest

test(e2e): migrate device auth health to vitest #9133

Workflow file for this run

# SPDX-FileCopyrightText: Copyright (c) 2026 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
# SPDX-License-Identifier: Apache-2.0
name: E2E / WSL
on:
workflow_dispatch:
pull_request:
paths:
- "bin/**"
- "nemoclaw/**"
- "scripts/**"
- "test/**"
- ".github/workflows/wsl-e2e.yaml"
- "package.json"
- "vitest.config.ts"
push:
branches:
- main
permissions:
contents: read
concurrency:
group: wsl-e2e-${{ github.ref }}
cancel-in-progress: true
jobs:
wsl-e2e:
runs-on: windows-latest
timeout-minutes: 90
env:
WSL_DISTRO: Ubuntu
NEMOCLAW_NON_INTERACTIVE: "1"
NEMOCLAW_ACCEPT_THIRD_PARTY_SOFTWARE: "1"
NEMOCLAW_RECREATE_SANDBOX: "1"
NEMOCLAW_SANDBOX_NAME: "e2e-wsl"
steps:
- name: Force LF line endings for checkout
shell: powershell
run: git config --global core.autocrlf false
- name: Checkout
uses: actions/checkout@df4cb1c069e1874edd31b4311f1884172cec0e10 # v6.0.3
- name: Resolve workspace paths for WSL
shell: powershell
run: |
$winPath = "${{ github.workspace }}"
$drive = $winPath.Substring(0,1).ToLower()
$rest = $winPath.Substring(2).Replace('\','/')
$wslCheckoutPath = "/mnt/$drive$rest"
$wslWorkdir = "/tmp/nemoclaw-wsl-workdir/${env:GITHUB_RUN_ID}-${env:GITHUB_RUN_ATTEMPT}"
"WSL_CHECKOUT_DIR=$wslCheckoutPath" | Out-File -FilePath $env:GITHUB_ENV -Encoding utf8 -Append
"WSL_WORKDIR=$wslWorkdir" | Out-File -FilePath $env:GITHUB_ENV -Encoding utf8 -Append
Write-Host "WSL_CHECKOUT_DIR=$wslCheckoutPath"
Write-Host "WSL_WORKDIR=$wslWorkdir"
- name: Ensure Ubuntu WSL exists
shell: powershell
run: |
wsl --list --verbose 2>&1 | Out-Default
# Native commands do not throw in PowerShell; check LASTEXITCODE.
$null = wsl -d $env:WSL_DISTRO -- echo ok 2>&1
if ($LASTEXITCODE -ne 0) {
$maxAttempts = 3
$installed = $false
for ($attempt = 1; $attempt -le $maxAttempts; $attempt++) {
Write-Host "Ubuntu not found - installing via wsl --install (attempt $attempt/$maxAttempts)"
wsl --install -d $env:WSL_DISTRO --no-launch --web-download
$installExitCode = $LASTEXITCODE
if ($installExitCode -eq 0) {
# The first launch initialises the distro with the default root user.
wsl -d $env:WSL_DISTRO -- bash -c 'echo distro initialised'
$launchExitCode = $LASTEXITCODE
if ($launchExitCode -eq 0) {
$installed = $true
break
}
Write-Warning "distro first-launch failed with exit code $launchExitCode"
} else {
Write-Warning "wsl --install failed with exit code $installExitCode"
}
# Some WSL installs return a non-zero code after registering a usable distro.
$null = wsl -d $env:WSL_DISTRO -- echo ok 2>&1
if ($LASTEXITCODE -eq 0) {
Write-Host 'Ubuntu became available after the install command returned non-zero'
$installed = $true
break
}
if ($attempt -lt $maxAttempts) {
Write-Host 'Cleaning up any partial WSL registration before retrying'
$null = wsl --unregister $env:WSL_DISTRO 2>&1
$delaySeconds = [Math]::Min(60, 20 * $attempt)
Write-Host "Retrying WSL install in $delaySeconds seconds..."
Start-Sleep -Seconds $delaySeconds
}
}
if (-not $installed) {
throw ("failed to install and initialize $env:WSL_DISTRO after $maxAttempts attempts")
}
} else {
Write-Host 'Ubuntu already available'
}
wsl --set-default $env:WSL_DISTRO
if ($LASTEXITCODE -ne 0) {
throw ('wsl --set-default failed with exit code ' + $LASTEXITCODE)
}
- name: Verify WSL
shell: powershell
run: |
wsl -d $env:WSL_DISTRO -- bash -lc "uname -a"
wsl -d $env:WSL_DISTRO -- bash -lc "cat /etc/os-release"
- name: Install Ubuntu dependencies
shell: powershell
run: |
$script = @'
set -euo pipefail
export DEBIAN_FRONTEND=noninteractive
printf '%s\n' \
'Acquire::ForceIPv4 "true";' \
'Acquire::Retries "5";' \
>/etc/apt/apt.conf.d/99github-actions-network
apt-get update
apt-get install -y bash ca-certificates curl git jq lsb-release make python3 python3-pip rsync tar unzip xz-utils
'@
$tmp = "$env:RUNNER_TEMP\wsl-step.sh"
[IO.File]::WriteAllText($tmp, ($script -replace "`r",""), (New-Object System.Text.UTF8Encoding $false))
$wslTmp = wsl -d $env:WSL_DISTRO -- wslpath -u ($tmp -replace '\\','/')
wsl -d $env:WSL_DISTRO -- bash -l $wslTmp
- name: Install Node.js 22 in WSL
shell: powershell
run: |
$script = @'
set -euo pipefail
curl -fsSL https://deb.nodesource.com/setup_22.x | bash -
apt-get install -y nodejs
node --version
npm --version
'@
$tmp = "$env:RUNNER_TEMP\wsl-step.sh"
[IO.File]::WriteAllText($tmp, ($script -replace "`r",""), (New-Object System.Text.UTF8Encoding $false))
$wslTmp = wsl -d $env:WSL_DISTRO -- wslpath -u ($tmp -replace '\\','/')
wsl -d $env:WSL_DISTRO -- bash -l $wslTmp
- name: Copy checkout into WSL ext4 workspace
shell: powershell
run: |
$checkout = $env:WSL_CHECKOUT_DIR
$workdir = $env:WSL_WORKDIR
$workdirParent = $workdir.Substring(0, $workdir.LastIndexOf('/'))
$script = @"
set -euo pipefail
echo 'Syncing checkout from $checkout to $workdir'
if [ ! -d '$checkout/.git' ]; then
echo 'Expected a Git checkout at $checkout' >&2
exit 1
fi
# Keep npm and test I/O on WSL's ext4 VHD. Running directly from
# /mnt/<drive> (DrvFS) is slower and has Windows-style permission
# semantics that hide Linux permission regressions.
rm -rf '$workdir'
mkdir -p '$workdirParent'
rsync -a --no-owner --no-group --delete \
--exclude '/node_modules/' \
--exclude '/nemoclaw/node_modules/' \
--exclude '/nemoclaw-blueprint/.venv/' \
'$checkout'/ '$workdir'/
git config --global --add safe.directory '$workdir'
git -C '$workdir' reset --hard HEAD
git -C '$workdir' clean -ffdx
git -C '$workdir' status --short
echo 'WSL ext4 workspace ready at $workdir'
"@
$tmp = "$env:RUNNER_TEMP\wsl-step.sh"
[IO.File]::WriteAllText($tmp, ($script -replace "`r",""), (New-Object System.Text.UTF8Encoding $false))
$wslTmp = wsl -d $env:WSL_DISTRO -- wslpath -u ($tmp -replace '\\','/')
wsl -d $env:WSL_DISTRO -- bash -l $wslTmp
- name: Install project dependencies and build plugin
shell: powershell
run: |
$script = @"
set -euo pipefail
cd '$env:WSL_WORKDIR'
npm install --ignore-scripts
npm run build:cli
cd nemoclaw
npm install --ignore-scripts
npm run build
"@
$tmp = "$env:RUNNER_TEMP\wsl-step.sh"
[IO.File]::WriteAllText($tmp, ($script -replace "`r",""), (New-Object System.Text.UTF8Encoding $false))
$wslTmp = wsl -d $env:WSL_DISTRO -- wslpath -u ($tmp -replace '\\','/')
wsl -d $env:WSL_DISTRO -- bash -l $wslTmp
- name: Detect Docker availability in WSL
id: docker
shell: powershell
run: |
$script = @'
if docker info >/dev/null 2>&1; then
echo DOCKER_OK=1
else
echo DOCKER_OK=0
fi
'@
$tmp = "$env:RUNNER_TEMP\wsl-step.sh"
[IO.File]::WriteAllText($tmp, ($script -replace "`r",""), (New-Object System.Text.UTF8Encoding $false))
$wslTmp = wsl -d $env:WSL_DISTRO -- wslpath -u ($tmp -replace '\\','/')
$result = wsl -d $env:WSL_DISTRO -- bash -l $wslTmp
if ($result -match 'DOCKER_OK=1') {
'docker_ok=true' | Out-File -FilePath $env:GITHUB_OUTPUT -Encoding utf8 -Append
Write-Host 'Docker is available in WSL'
} else {
'docker_ok=false' | Out-File -FilePath $env:GITHUB_OUTPUT -Encoding utf8 -Append
Write-Host 'Docker is not available in WSL; full E2E will be skipped'
}
- name: Run WSL full E2E
if: steps.docker.outputs.docker_ok == 'true'
shell: powershell
env:
NVIDIA_INFERENCE_API_KEY: ${{ secrets.NVIDIA_INFERENCE_API_KEY }}
GITHUB_TOKEN: ${{ github.token }}
run: |
$script = @"
set -euo pipefail
cd '$env:WSL_WORKDIR'
export NVIDIA_INFERENCE_API_KEY='$env:NVIDIA_INFERENCE_API_KEY'
export GITHUB_TOKEN='$env:GITHUB_TOKEN'
export NEMOCLAW_NON_INTERACTIVE='$env:NEMOCLAW_NON_INTERACTIVE'
export NEMOCLAW_ACCEPT_THIRD_PARTY_SOFTWARE='$env:NEMOCLAW_ACCEPT_THIRD_PARTY_SOFTWARE'
export NEMOCLAW_RECREATE_SANDBOX='$env:NEMOCLAW_RECREATE_SANDBOX'
export NEMOCLAW_SANDBOX_NAME='$env:NEMOCLAW_SANDBOX_NAME'
bash test/e2e/test-full-e2e.sh
"@
$tmp = "$env:RUNNER_TEMP\wsl-step.sh"
[IO.File]::WriteAllText($tmp, ($script -replace "`r",""), (New-Object System.Text.UTF8Encoding $false))
$wslTmp = wsl -d $env:WSL_DISTRO -- wslpath -u ($tmp -replace '\\','/')
wsl -d $env:WSL_DISTRO -- bash -l $wslTmp
- name: Explain skipped full E2E
if: steps.docker.outputs.docker_ok != 'true'
shell: powershell
run: |
Write-Host 'Skipping WSL full E2E because Docker is unavailable on this runner.'
Write-Host 'The workflow still validated the NemoClaw build flow inside Ubuntu WSL.'
- name: Upload install log on failure
if: failure()
uses: actions/upload-artifact@043fb46d1a93c77aae656e7c1c64a875d1fc6a0a # v7.0.1
with:
name: wsl-e2e-install-log
path: |
C:\Users\runneradmin\AppData\Local\Temp\nemoclaw-e2e-install.log
if-no-files-found: ignore