mirror of
https://github.com/glittercowboy/get-shit-done
synced 2026-04-25 17:25:23 +02:00
Three install code paths were leaking Claude-specific references into Qwen installs: copyCommandsAsClaudeSkills lacked runtime-aware content replacement, the agents copy loop had no isQwen branch, and the hooks template loop only replaced the quoted '.claude' form. Added CLAUDE.md, Claude Code, and .claude/ replacements across all three paths plus copyWithPathReplacement's Qwen .md branch. Includes regression test that walks the full .qwen/ tree after install and asserts zero Claude references outside CHANGELOG.md. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
317 lines
10 KiB
JavaScript
317 lines
10 KiB
JavaScript
process.env.GSD_TEST_MODE = '1';
|
|
|
|
const { test, describe, beforeEach, afterEach } = require('node:test');
|
|
const assert = require('node:assert/strict');
|
|
const fs = require('node:fs');
|
|
const path = require('node:path');
|
|
const os = require('node:os');
|
|
const { createTempDir, cleanup } = require('./helpers.cjs');
|
|
|
|
const {
|
|
getDirName,
|
|
getGlobalDir,
|
|
getConfigDirFromHome,
|
|
install,
|
|
uninstall,
|
|
writeManifest,
|
|
} = require('../bin/install.js');
|
|
|
|
describe('Qwen Code runtime directory mapping', () => {
|
|
test('maps Qwen to .qwen for local installs', () => {
|
|
assert.strictEqual(getDirName('qwen'), '.qwen');
|
|
});
|
|
|
|
test('maps Qwen to ~/.qwen for global installs', () => {
|
|
assert.strictEqual(getGlobalDir('qwen'), path.join(os.homedir(), '.qwen'));
|
|
});
|
|
|
|
test('returns .qwen config fragments for local and global installs', () => {
|
|
assert.strictEqual(getConfigDirFromHome('qwen', false), "'.qwen'");
|
|
assert.strictEqual(getConfigDirFromHome('qwen', true), "'.qwen'");
|
|
});
|
|
});
|
|
|
|
describe('getGlobalDir (Qwen Code)', () => {
|
|
let originalQwenConfigDir;
|
|
|
|
beforeEach(() => {
|
|
originalQwenConfigDir = process.env.QWEN_CONFIG_DIR;
|
|
});
|
|
|
|
afterEach(() => {
|
|
if (originalQwenConfigDir !== undefined) {
|
|
process.env.QWEN_CONFIG_DIR = originalQwenConfigDir;
|
|
} else {
|
|
delete process.env.QWEN_CONFIG_DIR;
|
|
}
|
|
});
|
|
|
|
test('returns ~/.qwen with no env var or explicit dir', () => {
|
|
delete process.env.QWEN_CONFIG_DIR;
|
|
const result = getGlobalDir('qwen');
|
|
assert.strictEqual(result, path.join(os.homedir(), '.qwen'));
|
|
});
|
|
|
|
test('returns explicit dir when provided', () => {
|
|
const result = getGlobalDir('qwen', '/custom/qwen-path');
|
|
assert.strictEqual(result, '/custom/qwen-path');
|
|
});
|
|
|
|
test('respects QWEN_CONFIG_DIR env var', () => {
|
|
process.env.QWEN_CONFIG_DIR = '~/custom-qwen';
|
|
const result = getGlobalDir('qwen');
|
|
assert.strictEqual(result, path.join(os.homedir(), 'custom-qwen'));
|
|
});
|
|
|
|
test('explicit dir takes priority over QWEN_CONFIG_DIR', () => {
|
|
process.env.QWEN_CONFIG_DIR = '~/from-env';
|
|
const result = getGlobalDir('qwen', '/explicit/path');
|
|
assert.strictEqual(result, '/explicit/path');
|
|
});
|
|
|
|
test('does not break other runtimes', () => {
|
|
assert.strictEqual(getGlobalDir('claude'), path.join(os.homedir(), '.claude'));
|
|
assert.strictEqual(getGlobalDir('codex'), path.join(os.homedir(), '.codex'));
|
|
});
|
|
});
|
|
|
|
describe('Qwen Code local install/uninstall', () => {
|
|
let tmpDir;
|
|
let previousCwd;
|
|
|
|
beforeEach(() => {
|
|
tmpDir = createTempDir('gsd-qwen-install-');
|
|
previousCwd = process.cwd();
|
|
process.chdir(tmpDir);
|
|
});
|
|
|
|
afterEach(() => {
|
|
process.chdir(previousCwd);
|
|
cleanup(tmpDir);
|
|
});
|
|
|
|
test('installs GSD into ./.qwen and removes it cleanly', () => {
|
|
const result = install(false, 'qwen');
|
|
const targetDir = path.join(tmpDir, '.qwen');
|
|
|
|
assert.strictEqual(result.runtime, 'qwen');
|
|
assert.strictEqual(result.configDir, fs.realpathSync(targetDir));
|
|
|
|
assert.ok(fs.existsSync(path.join(targetDir, 'skills', 'gsd-help', 'SKILL.md')));
|
|
assert.ok(fs.existsSync(path.join(targetDir, 'get-shit-done', 'VERSION')));
|
|
assert.ok(fs.existsSync(path.join(targetDir, 'agents')));
|
|
|
|
const manifest = writeManifest(targetDir, 'qwen');
|
|
assert.ok(Object.keys(manifest.files).some(file => file.startsWith('skills/gsd-help/')), manifest);
|
|
|
|
uninstall(false, 'qwen');
|
|
|
|
assert.ok(!fs.existsSync(path.join(targetDir, 'skills', 'gsd-help')), 'Qwen skill directory removed');
|
|
assert.ok(!fs.existsSync(path.join(targetDir, 'get-shit-done')), 'get-shit-done removed');
|
|
});
|
|
});
|
|
|
|
describe('E2E: Qwen Code uninstall skills cleanup', () => {
|
|
let tmpDir;
|
|
let previousCwd;
|
|
|
|
beforeEach(() => {
|
|
tmpDir = createTempDir('gsd-qwen-uninstall-');
|
|
previousCwd = process.cwd();
|
|
process.chdir(tmpDir);
|
|
});
|
|
|
|
afterEach(() => {
|
|
process.chdir(previousCwd);
|
|
cleanup(tmpDir);
|
|
});
|
|
|
|
test('removes all gsd-* skill directories on --qwen --uninstall', () => {
|
|
const targetDir = path.join(tmpDir, '.qwen');
|
|
install(false, 'qwen');
|
|
|
|
const skillsDir = path.join(targetDir, 'skills');
|
|
assert.ok(fs.existsSync(skillsDir), 'skills dir exists after install');
|
|
|
|
const installedSkills = fs.readdirSync(skillsDir, { withFileTypes: true })
|
|
.filter(e => e.isDirectory() && e.name.startsWith('gsd-'));
|
|
assert.ok(installedSkills.length > 0, `found ${installedSkills.length} gsd-* skill dirs before uninstall`);
|
|
|
|
uninstall(false, 'qwen');
|
|
|
|
if (fs.existsSync(skillsDir)) {
|
|
const remainingGsd = fs.readdirSync(skillsDir, { withFileTypes: true })
|
|
.filter(e => e.isDirectory() && e.name.startsWith('gsd-'));
|
|
assert.strictEqual(remainingGsd.length, 0,
|
|
`Expected 0 gsd-* skill dirs after uninstall, found: ${remainingGsd.map(e => e.name).join(', ')}`);
|
|
}
|
|
});
|
|
|
|
test('preserves non-GSD skill directories during --qwen --uninstall', () => {
|
|
const targetDir = path.join(tmpDir, '.qwen');
|
|
install(false, 'qwen');
|
|
|
|
const customSkillDir = path.join(targetDir, 'skills', 'my-custom-skill');
|
|
fs.mkdirSync(customSkillDir, { recursive: true });
|
|
fs.writeFileSync(path.join(customSkillDir, 'SKILL.md'), '# My Custom Skill\n');
|
|
|
|
assert.ok(fs.existsSync(path.join(customSkillDir, 'SKILL.md')), 'custom skill exists before uninstall');
|
|
|
|
uninstall(false, 'qwen');
|
|
|
|
assert.ok(fs.existsSync(path.join(customSkillDir, 'SKILL.md')),
|
|
'Non-GSD skill directory should be preserved after Qwen uninstall');
|
|
});
|
|
|
|
test('removes engine directory on --qwen --uninstall', () => {
|
|
const targetDir = path.join(tmpDir, '.qwen');
|
|
install(false, 'qwen');
|
|
|
|
assert.ok(fs.existsSync(path.join(targetDir, 'get-shit-done', 'VERSION')),
|
|
'engine exists before uninstall');
|
|
|
|
uninstall(false, 'qwen');
|
|
|
|
assert.ok(!fs.existsSync(path.join(targetDir, 'get-shit-done')),
|
|
'get-shit-done engine should be removed after Qwen uninstall');
|
|
});
|
|
});
|
|
|
|
// ─── Regression: no Claude references leak into Qwen install (#2112) ──────────
|
|
|
|
describe('Qwen install contains no leaked Claude references (#2112)', () => {
|
|
let tmpDir;
|
|
let previousCwd;
|
|
|
|
beforeEach(() => {
|
|
tmpDir = createTempDir('gsd-qwen-refs-');
|
|
previousCwd = process.cwd();
|
|
process.chdir(tmpDir);
|
|
install(false, 'qwen');
|
|
});
|
|
|
|
afterEach(() => {
|
|
process.chdir(previousCwd);
|
|
cleanup(tmpDir);
|
|
});
|
|
|
|
/**
|
|
* Recursively walk a directory and return all file paths.
|
|
*/
|
|
function walk(dir) {
|
|
const results = [];
|
|
for (const entry of fs.readdirSync(dir, { withFileTypes: true })) {
|
|
const full = path.join(dir, entry.name);
|
|
if (entry.isDirectory()) {
|
|
results.push(...walk(full));
|
|
} else {
|
|
results.push(full);
|
|
}
|
|
}
|
|
return results;
|
|
}
|
|
|
|
/**
|
|
* Return files under .qwen/ that contain Claude references,
|
|
* excluding CHANGELOG.md (historical accuracy) and VERSION (no prose).
|
|
*/
|
|
function findClaudeLeaks() {
|
|
const qwenDir = path.join(tmpDir, '.qwen');
|
|
const allFiles = walk(qwenDir);
|
|
const textFiles = allFiles.filter(f =>
|
|
f.endsWith('.md') || f.endsWith('.cjs') || f.endsWith('.js')
|
|
);
|
|
const excluded = ['CHANGELOG.md'];
|
|
const candidates = textFiles.filter(f =>
|
|
!excluded.includes(path.basename(f))
|
|
);
|
|
const leaks = [];
|
|
for (const file of candidates) {
|
|
const content = fs.readFileSync(file, 'utf8');
|
|
if (/\bCLAUDE\.md\b/.test(content) ||
|
|
/\bClaude Code\b/.test(content) ||
|
|
/\.claude\//.test(content)) {
|
|
leaks.push(path.relative(tmpDir, file));
|
|
}
|
|
}
|
|
return leaks;
|
|
}
|
|
|
|
test('skills contain no CLAUDE.md or Claude Code references', () => {
|
|
const qwenDir = path.join(tmpDir, '.qwen');
|
|
const skillsDir = path.join(qwenDir, 'skills');
|
|
assert.ok(fs.existsSync(skillsDir), 'skills directory exists');
|
|
|
|
const skillFiles = walk(skillsDir).filter(f => f.endsWith('.md'));
|
|
assert.ok(skillFiles.length > 0, 'at least one skill file exists');
|
|
|
|
const leaks = [];
|
|
for (const file of skillFiles) {
|
|
const content = fs.readFileSync(file, 'utf8');
|
|
if (/\bCLAUDE\.md\b/.test(content) || /\bClaude Code\b/.test(content)) {
|
|
leaks.push(path.relative(tmpDir, file));
|
|
}
|
|
}
|
|
assert.strictEqual(leaks.length, 0,
|
|
[
|
|
'Skills should not contain Claude references after Qwen install.',
|
|
'Leaking files:',
|
|
...leaks,
|
|
].join('\n'));
|
|
});
|
|
|
|
test('agents contain no CLAUDE.md or Claude Code references', () => {
|
|
const agentsDir = path.join(tmpDir, '.qwen', 'agents');
|
|
assert.ok(fs.existsSync(agentsDir), 'agents directory exists');
|
|
|
|
const agentFiles = walk(agentsDir).filter(f => f.endsWith('.md'));
|
|
assert.ok(agentFiles.length > 0, 'at least one agent file exists');
|
|
|
|
const leaks = [];
|
|
for (const file of agentFiles) {
|
|
const content = fs.readFileSync(file, 'utf8');
|
|
if (/\bCLAUDE\.md\b/.test(content) || /\bClaude Code\b/.test(content)) {
|
|
leaks.push(path.relative(tmpDir, file));
|
|
}
|
|
}
|
|
assert.strictEqual(leaks.length, 0,
|
|
[
|
|
'Agents should not contain Claude references after Qwen install.',
|
|
'Leaking files:',
|
|
...leaks,
|
|
].join('\n'));
|
|
});
|
|
|
|
test('hooks contain no .claude/ path references', () => {
|
|
const hooksDir = path.join(tmpDir, '.qwen', 'hooks');
|
|
if (!fs.existsSync(hooksDir)) {
|
|
return; // hooks may not be present in local installs
|
|
}
|
|
|
|
const hookFiles = walk(hooksDir).filter(f => f.endsWith('.js'));
|
|
const leaks = [];
|
|
for (const file of hookFiles) {
|
|
const content = fs.readFileSync(file, 'utf8');
|
|
if (/\.claude\//.test(content)) {
|
|
leaks.push(path.relative(tmpDir, file));
|
|
}
|
|
}
|
|
assert.strictEqual(leaks.length, 0,
|
|
[
|
|
'Hooks should not contain .claude/ path references after Qwen install.',
|
|
'Leaking files:',
|
|
...leaks,
|
|
].join('\n'));
|
|
});
|
|
|
|
test('full tree scan finds zero Claude references outside CHANGELOG.md', () => {
|
|
const leaks = findClaudeLeaks();
|
|
assert.strictEqual(leaks.length, 0,
|
|
[
|
|
'No files under .qwen/ (except CHANGELOG.md) should contain Claude references.',
|
|
`Found ${leaks.length} leaking file(s):`,
|
|
...leaks,
|
|
].join('\n'));
|
|
});
|
|
});
|