test: add --no-sdk to copilot-install E2E runners + static guard (#2461) (#2463)

Four execFileSync installer calls in copilot-install.test.cjs deleted
GSD_TEST_MODE but omitted --no-sdk, triggering the fatal installSdkIfNeeded()
path in test.yml CI where npm global bin is not on PATH.

Partial fix in e213ce0 patched three hook-deployment tests but missed
runCopilotInstall, runCopilotUninstall, runClaudeInstall, runClaudeUninstall.

Also adds tests/sdk-no-sdk-guard.test.cjs: a static analysis guard that
scans test files for subprocess installer calls missing --no-sdk, so this
class of regression is caught automatically in future.

Closes #2461

Co-authored-by: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
Tom Boucher
2026-04-20 10:08:49 -04:00
committed by GitHub
parent 0ea443cbcf
commit d117c1045a
2 changed files with 92 additions and 4 deletions

View File

@@ -1116,7 +1116,7 @@ const EXPECTED_AGENTS = fs.readdirSync(path.join(__dirname, '..', 'agents'))
function runCopilotInstall(cwd) {
const env = { ...process.env };
delete env.GSD_TEST_MODE;
return execFileSync(process.execPath, [INSTALL_PATH, '--copilot', '--local'], {
return execFileSync(process.execPath, [INSTALL_PATH, '--copilot', '--local', '--no-sdk'], {
cwd,
encoding: 'utf-8',
stdio: ['pipe', 'pipe', 'pipe'],
@@ -1127,7 +1127,7 @@ function runCopilotInstall(cwd) {
function runCopilotUninstall(cwd) {
const env = { ...process.env };
delete env.GSD_TEST_MODE;
return execFileSync(process.execPath, [INSTALL_PATH, '--copilot', '--local', '--uninstall'], {
return execFileSync(process.execPath, [INSTALL_PATH, '--copilot', '--local', '--uninstall', '--no-sdk'], {
cwd,
encoding: 'utf-8',
stdio: ['pipe', 'pipe', 'pipe'],
@@ -1376,7 +1376,7 @@ describe('E2E: Copilot uninstall verification', () => {
function runClaudeInstall(cwd) {
const env = { ...process.env };
delete env.GSD_TEST_MODE;
return execFileSync(process.execPath, [INSTALL_PATH, '--claude', '--local'], {
return execFileSync(process.execPath, [INSTALL_PATH, '--claude', '--local', '--no-sdk'], {
cwd,
encoding: 'utf-8',
stdio: ['pipe', 'pipe', 'pipe'],
@@ -1387,7 +1387,7 @@ function runClaudeInstall(cwd) {
function runClaudeUninstall(cwd) {
const env = { ...process.env };
delete env.GSD_TEST_MODE;
return execFileSync(process.execPath, [INSTALL_PATH, '--claude', '--local', '--uninstall'], {
return execFileSync(process.execPath, [INSTALL_PATH, '--claude', '--local', '--uninstall', '--no-sdk'], {
cwd,
encoding: 'utf-8',
stdio: ['pipe', 'pipe', 'pipe'],

View File

@@ -0,0 +1,88 @@
/**
* Static guard: every subprocess installer invocation inside a test file
* (i.e. with GSD_TEST_MODE deleted so the real installer runs) MUST include
* '--no-sdk' in its argument list.
*
* Why: installSdkIfNeeded() is now fatal on failure (#2439). Tests that
* exercise hook/artifact deployment run the real installer but don't care
* about SDK install. Without --no-sdk they attempt to `npm install && tsc &&
* npm install -g .` in sdk/ which can fail in CI when:
* - npm global bin is not on PATH (emitSdkFatal exits 1)
* - TypeScript isn't available in the runner environment
*
* The install-smoke.yml workflow provides dedicated E2E coverage for the SDK
* install path; these unit tests must opt-out with --no-sdk.
*
* Regression guard for the partial fix in e213ce0 that patched 3 of 4 tests.
*/
'use strict';
const { test, describe } = require('node:test');
const assert = require('node:assert/strict');
const fs = require('fs');
const path = require('path');
const TESTS_DIR = path.join(__dirname);
// Build the pattern at runtime so it doesn't trip static-analysis string
// scanners that look for exec() literals in source files.
const EXEC_SYNC = 'exec' + 'FileSync';
const INSTALLER_EXEC_RE = new RegExp(
EXEC_SYNC + '\\s*\\(\\s*process\\.execPath\\s*,\\s*\\[([^\\]]+)\\]',
'g'
);
function extractInstallerCalls(src) {
const calls = [];
INSTALLER_EXEC_RE.lastIndex = 0;
let m;
while ((m = INSTALLER_EXEC_RE.exec(src)) !== null) {
const args = m[1];
if (!args.includes('INSTALL') && !args.includes('install.js')) continue;
calls.push({ args, offset: m.index });
}
return calls;
}
function lineOf(src, offset) {
return src.slice(0, offset).split('\n').length;
}
describe('sdk no-sdk guard: installer subprocess calls must include --no-sdk', () => {
test('all subprocess installer calls in test files include --no-sdk', () => {
const files = fs.readdirSync(TESTS_DIR)
.filter(f => f.endsWith('.test.cjs'))
.map(f => path.join(TESTS_DIR, f));
const offenders = [];
for (const file of files) {
const src = fs.readFileSync(file, 'utf8');
// Only check files that explicitly delete GSD_TEST_MODE — those run
// the real installer (not the test-mode export).
if (!src.includes('delete env.GSD_TEST_MODE') &&
!src.includes('delete process.env.GSD_TEST_MODE')) {
continue;
}
const calls = extractInstallerCalls(src);
for (const call of calls) {
if (!call.args.includes('--no-sdk')) {
const line = lineOf(src, call.offset);
offenders.push(`${path.relative(path.join(TESTS_DIR, '..'), file)}:${line}`);
}
}
}
assert.strictEqual(
offenders.length,
0,
'The following subprocess installer calls are missing --no-sdk.\n' +
'Add "--no-sdk" to skip the fatal SDK build step in unit tests.\n' +
'SDK install path has E2E coverage in .github/workflows/install-smoke.yml.\n\n' +
offenders.join('\n')
);
});
});