| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409 |
- import { describe, it, expect, beforeEach, afterEach } from 'vitest';
- import * as fs from 'node:fs';
- import * as path from 'node:path';
- import * as os from 'node:os';
- import {
- detectInstallMethod,
- deriveInstallDir,
- parseSemver,
- compareVersions,
- isUpdateAvailable,
- normalizeVersion,
- stripV,
- parseLatestTagFromLocation,
- reindexAdvisory,
- runUpgrade,
- buildWindowsUpgradeScript,
- NPM_PACKAGE,
- type InstallMethod,
- type UpgradeDeps,
- } from '../src/upgrade';
- import { EXTRACTION_VERSION } from '../src/extraction/extraction-version';
- import { CodeGraph } from '../src';
- // ---------------------------------------------------------------------------
- // detectInstallMethod — structural detection from the running file's path
- // ---------------------------------------------------------------------------
- describe('detectInstallMethod', () => {
- // A bundle exists if a vendored node + launcher sit next to lib/.
- function bundleExists(present: Set<string>) {
- return (p: string) => present.has(p.replace(/\\/g, '/'));
- }
- it('detects a unix bundle and derives the install dir from the versions/ layout', () => {
- const root = '/home/u/.codegraph/versions/v0.9.9';
- const filename = `${root}/lib/dist/bin/codegraph.js`;
- const present = new Set([`${root}/node`, `${root}/bin/codegraph`, '/home/u/.codegraph']);
- const m = detectInstallMethod({
- filename,
- platform: 'linux',
- cwd: '/home/u/project',
- exists: bundleExists(present),
- });
- expect(m).toEqual({
- kind: 'bundle',
- os: 'unix',
- bundleRoot: root,
- installDir: '/home/u/.codegraph',
- });
- });
- it('detects a windows bundle and derives the install dir from current\\', () => {
- const root = 'C:/Users/u/AppData/Local/codegraph/current';
- const filename = `${root}/lib/dist/bin/codegraph.js`;
- const present = new Set([`${root}/node.exe`, `${root}/bin/codegraph.cmd`]);
- const m = detectInstallMethod({
- filename,
- platform: 'win32',
- cwd: 'C:/Users/u/project',
- exists: bundleExists(present),
- }) as Extract<InstallMethod, { kind: 'bundle' }>;
- expect(m.kind).toBe('bundle');
- expect(m.os).toBe('windows');
- // win32 path math emits backslashes; compare separator-independently.
- expect(m.installDir?.replace(/\\/g, '/')).toBe('C:/Users/u/AppData/Local/codegraph');
- });
- it('detects a global npm install', () => {
- const filename = '/usr/local/lib/node_modules/@colbymchenry/codegraph/dist/bin/codegraph.js';
- const m = detectInstallMethod({
- filename,
- platform: 'linux',
- cwd: '/home/u/project',
- exists: () => false,
- });
- expect(m).toEqual({ kind: 'npm', scope: 'global' });
- });
- it('detects a local (project) npm install as local', () => {
- const cwd = '/home/u/project';
- const filename = `${cwd}/node_modules/@colbymchenry/codegraph/dist/bin/codegraph.js`;
- const m = detectInstallMethod({ filename, platform: 'linux', cwd, exists: () => false });
- expect(m).toEqual({ kind: 'npm', scope: 'local' });
- });
- it('detects an npx run from the _npx cache', () => {
- const filename = '/home/u/.npm/_npx/abc123/node_modules/@colbymchenry/codegraph/dist/bin/codegraph.js';
- const m = detectInstallMethod({ filename, platform: 'linux', cwd: '/home/u', exists: () => false });
- expect(m).toEqual({ kind: 'npx' });
- });
- it('detects a source checkout via sibling package.json + .git', () => {
- const repo = '/home/u/dev/codegraph';
- const filename = `${repo}/dist/bin/codegraph.js`;
- const present = new Set([`${repo}/package.json`, `${repo}/.git`]);
- const m = detectInstallMethod({
- filename,
- platform: 'darwin',
- cwd: repo,
- exists: bundleExists(present),
- });
- expect(m).toEqual({ kind: 'source', root: repo });
- });
- it('returns unknown for an unrecognized layout', () => {
- const m = detectInstallMethod({
- filename: '/opt/weird/place/codegraph.js',
- platform: 'linux',
- cwd: '/tmp',
- exists: () => false,
- });
- expect(m.kind).toBe('unknown');
- });
- });
- describe('deriveInstallDir', () => {
- it('unix: returns the dir above versions/', () => {
- expect(deriveInstallDir('/a/b/.codegraph/versions/v1.2.3', 'unix', () => true)).toBe('/a/b/.codegraph');
- });
- it('unix: null when not under versions/', () => {
- expect(deriveInstallDir('/a/b/somewhere', 'unix', () => true)).toBeNull();
- });
- it('windows: returns the parent of current\\', () => {
- expect(deriveInstallDir('C:/x/codegraph/current', 'windows', () => true)?.replace(/\\/g, '/')).toBe('C:/x/codegraph');
- });
- it('windows: null when basename is not current', () => {
- expect(deriveInstallDir('C:/x/codegraph/v1', 'windows', () => true)).toBeNull();
- });
- });
- // ---------------------------------------------------------------------------
- // version helpers
- // ---------------------------------------------------------------------------
- describe('version helpers', () => {
- it('parseSemver handles v-prefix and prerelease', () => {
- expect(parseSemver('v1.2.3')).toEqual({ major: 1, minor: 2, patch: 3, pre: null });
- expect(parseSemver('1.2.3-rc.1')).toEqual({ major: 1, minor: 2, patch: 3, pre: 'rc.1' });
- expect(parseSemver('not-a-version')).toBeNull();
- });
- it('compareVersions orders correctly incl. prerelease < release', () => {
- expect(compareVersions('1.0.1', '1.0.0')).toBeGreaterThan(0);
- expect(compareVersions('1.0.0', '1.1.0')).toBeLessThan(0);
- expect(compareVersions('v2.0.0', '2.0.0')).toBe(0);
- expect(compareVersions('1.0.0-rc.1', '1.0.0')).toBeLessThan(0);
- });
- it('isUpdateAvailable compares, and falls back to string-inequality for unparseable', () => {
- expect(isUpdateAvailable('0.9.8', '0.9.9')).toBe(true);
- expect(isUpdateAvailable('0.9.9', '0.9.9')).toBe(false);
- expect(isUpdateAvailable('0.9.9', '0.9.8')).toBe(false);
- // dev sentinel can't parse → any difference means "update available"
- expect(isUpdateAvailable('0.0.0-unknown', '0.9.9')).toBe(true);
- });
- it('normalizeVersion / stripV round-trip', () => {
- expect(normalizeVersion('0.9.9')).toBe('v0.9.9');
- expect(normalizeVersion('v0.9.9')).toBe('v0.9.9');
- expect(stripV('v0.9.9')).toBe('0.9.9');
- expect(stripV('0.9.9')).toBe('0.9.9');
- });
- it('parseLatestTagFromLocation extracts the tag from a releases redirect', () => {
- expect(parseLatestTagFromLocation('https://github.com/colbymchenry/codegraph/releases/tag/v0.9.9')).toBe('v0.9.9');
- expect(parseLatestTagFromLocation('https://github.com/o/r/releases/tag/v1.2.3?foo=bar')).toBe('v1.2.3');
- expect(parseLatestTagFromLocation(undefined)).toBeNull();
- expect(parseLatestTagFromLocation('https://github.com/o/r/releases')).toBeNull();
- });
- it('reindexAdvisory mentions the refresh commands', () => {
- const a = reindexAdvisory();
- expect(a).toContain('codegraph sync');
- expect(a).toContain('codegraph index -f');
- });
- it('buildWindowsUpgradeScript targets the right asset per arch and renames-not-deletes the exe', () => {
- const arm = buildWindowsUpgradeScript('C:\\cg\\current', 'v1.2.3', 'arm64');
- expect(arm).toContain('releases/download/v1.2.3/codegraph-win32-arm64.zip');
- expect(arm).toContain("$dest='C:\\cg\\current'");
- expect(arm).toContain('Rename-Item'); // never Remove-Item on the locked exe
- expect(arm).not.toMatch(/Remove-Item[^;]*\$dest'?\s*;/); // doesn't delete current\
- const x64 = buildWindowsUpgradeScript('C:\\cg\\current', 'v1.2.3', 'x64');
- expect(x64).toContain('codegraph-win32-x64.zip');
- });
- });
- // ---------------------------------------------------------------------------
- // runUpgrade orchestration — mocked side-effects
- // ---------------------------------------------------------------------------
- interface Calls {
- runs: Array<{ cmd: string; args: string[]; env?: NodeJS.ProcessEnv }>;
- logs: string[];
- errors: string[];
- }
- function makeDeps(
- overrides: Partial<UpgradeDeps> & { method: InstallMethod; currentVersion: string },
- runExit = 0
- ): { deps: UpgradeDeps; calls: Calls } {
- const calls: Calls = { runs: [], logs: [], errors: [] };
- const deps: UpgradeDeps = {
- currentVersion: overrides.currentVersion,
- method: overrides.method,
- resolveLatest: overrides.resolveLatest ?? (async () => 'v0.9.9'),
- run: (cmd, args, env) => {
- calls.runs.push({ cmd, args, env });
- return runExit;
- },
- hasCommand: overrides.hasCommand ?? ((c) => c === 'curl'),
- log: (m) => calls.logs.push(m),
- warn: (m) => calls.logs.push(m),
- error: (m) => calls.errors.push(m),
- platform: overrides.platform ?? 'linux',
- };
- return { deps, calls };
- }
- /** Decode a `-EncodedCommand` base64 (UTF-16LE) payload back to its script. */
- function decodeEncodedCommand(args: string[]): string {
- const i = args.indexOf('-EncodedCommand');
- if (i < 0) throw new Error('no -EncodedCommand in args');
- return Buffer.from(args[i + 1]!, 'base64').toString('utf16le');
- }
- describe('runUpgrade', () => {
- it('does nothing when already up to date', async () => {
- const { deps, calls } = makeDeps({ method: { kind: 'npm', scope: 'global' }, currentVersion: '0.9.9' });
- const code = await runUpgrade({}, deps);
- expect(code).toBe(0);
- expect(calls.runs).toHaveLength(0);
- expect(calls.logs.join('\n')).toMatch(/up to date/i);
- });
- it('--check reports an available update without running anything', async () => {
- const { deps, calls } = makeDeps({
- method: { kind: 'npm', scope: 'global' },
- currentVersion: '0.9.8',
- });
- const code = await runUpgrade({ check: true }, deps);
- expect(code).toBe(0);
- expect(calls.runs).toHaveLength(0);
- expect(calls.logs.join('\n')).toMatch(/update is available/i);
- });
- it('unix bundle: runs the installer via sh with the derived install dir', async () => {
- const { deps, calls } = makeDeps({
- method: { kind: 'bundle', os: 'unix', bundleRoot: '/h/.codegraph/versions/v0.9.8', installDir: '/h/.codegraph' },
- currentVersion: '0.9.8',
- });
- const code = await runUpgrade({}, deps);
- expect(code).toBe(0);
- expect(calls.runs).toHaveLength(1);
- expect(calls.runs[0].cmd).toBe('sh');
- expect(calls.runs[0].args[0]).toBe('-c');
- expect(calls.runs[0].args[1]).toContain('curl -fsSL');
- expect(calls.runs[0].args[1]).toContain('| sh');
- expect(calls.runs[0].env?.CODEGRAPH_INSTALL_DIR).toBe('/h/.codegraph');
- expect(calls.logs.join('\n')).toMatch(/codegraph sync/); // re-index advisory printed
- });
- it('unix bundle: falls back to wget, and errors when neither downloader exists', async () => {
- const { deps, calls } = makeDeps({
- method: { kind: 'bundle', os: 'unix', bundleRoot: '/h/.codegraph/versions/v0.9.8', installDir: null },
- currentVersion: '0.9.8',
- hasCommand: () => false,
- });
- const code = await runUpgrade({}, deps);
- expect(code).toBe(1);
- expect(calls.runs).toHaveLength(0);
- expect(calls.errors.join('\n')).toMatch(/curl nor wget/i);
- });
- it('windows bundle: runs a synchronous in-place (rename + extract) powershell upgrade', async () => {
- const { deps, calls } = makeDeps({
- method: { kind: 'bundle', os: 'windows', bundleRoot: 'C:/x/codegraph/current', installDir: 'C:/x/codegraph' },
- currentVersion: '0.9.8',
- platform: 'win32',
- });
- const code = await runUpgrade({}, deps);
- expect(code).toBe(0);
- expect(calls.runs).toHaveLength(1);
- expect(calls.runs[0].cmd).toBe('powershell.exe');
- const decoded = decodeEncodedCommand(calls.runs[0].args);
- // Downloads the right asset, renames the locked exe aside, copies over current\.
- expect(decoded).toContain('releases/download/v0.9.9/codegraph-win32-');
- expect(decoded).toContain('Rename-Item');
- expect(decoded).toContain('node.exe.old-');
- expect(decoded).toContain('Copy-Item');
- });
- it('windows bundle: a non-zero installer exit is a failure', async () => {
- const { deps, calls } = makeDeps(
- {
- method: { kind: 'bundle', os: 'windows', bundleRoot: 'C:/x/codegraph/current', installDir: 'C:/x/codegraph' },
- currentVersion: '0.9.8',
- platform: 'win32',
- },
- 1
- );
- const code = await runUpgrade({}, deps);
- expect(code).toBe(1);
- expect(calls.errors.join('\n')).toMatch(/exited with code/i);
- });
- it('npm global: shells out to npm install -g @pkg@latest', async () => {
- const { deps, calls } = makeDeps({
- method: { kind: 'npm', scope: 'global' },
- currentVersion: '0.9.8',
- });
- const code = await runUpgrade({}, deps);
- expect(code).toBe(0);
- expect(calls.runs[0].cmd).toBe('npm');
- expect(calls.runs[0].args).toEqual(['install', '-g', `${NPM_PACKAGE}@latest`]);
- });
- it('npm on win32 uses npm.cmd', async () => {
- const { deps, calls } = makeDeps({
- method: { kind: 'npm', scope: 'global' },
- currentVersion: '0.9.8',
- platform: 'win32',
- });
- await runUpgrade({}, deps);
- expect(calls.runs[0].cmd).toBe('npm.cmd');
- });
- it('npm: a pinned version is passed through as @<version>', async () => {
- const { deps, calls } = makeDeps({
- method: { kind: 'npm', scope: 'global' },
- currentVersion: '0.9.9',
- });
- await runUpgrade({ version: '0.9.8' }, deps);
- // npm spec carries no leading "v".
- expect(calls.runs[0].args).toEqual(['install', '-g', `${NPM_PACKAGE}@0.9.8`]);
- });
- it('npm: surfaces a non-zero exit as failure', async () => {
- const { deps, calls } = makeDeps(
- { method: { kind: 'npm', scope: 'global' }, currentVersion: '0.9.8' },
- 1
- );
- const code = await runUpgrade({}, deps);
- expect(code).toBe(1);
- expect(calls.errors.join('\n')).toMatch(/npm exited/i);
- });
- it('npx: nothing to upgrade', async () => {
- const { deps, calls } = makeDeps({ method: { kind: 'npx' }, currentVersion: '0.9.8' });
- const code = await runUpgrade({}, deps);
- expect(code).toBe(0);
- expect(calls.runs).toHaveLength(0);
- expect(calls.logs.join('\n')).toMatch(/nothing to upgrade/i);
- });
- it('source: tells the user to git pull, runs nothing', async () => {
- const { deps, calls } = makeDeps({
- method: { kind: 'source', root: '/dev/codegraph' },
- currentVersion: '0.9.8',
- });
- const code = await runUpgrade({}, deps);
- expect(code).toBe(0);
- expect(calls.runs).toHaveLength(0);
- expect(calls.logs.join('\n')).toMatch(/git pull/);
- });
- });
- // ---------------------------------------------------------------------------
- // Re-index staleness — real index, real metadata stamp
- // ---------------------------------------------------------------------------
- describe('index extraction-version stamp / isIndexStale', () => {
- let dir: string;
- beforeEach(() => {
- dir = fs.mkdtempSync(path.join(os.tmpdir(), 'cg-upgrade-stamp-'));
- });
- afterEach(() => {
- fs.rmSync(dir, { recursive: true, force: true });
- });
- it('stamps the current extraction version on full index and is not stale', async () => {
- fs.writeFileSync(path.join(dir, 'a.ts'), 'export function hello() { return 1; }\n');
- const cg = await CodeGraph.init(dir, { index: false });
- // No index yet → not stale (nothing to refresh).
- expect(cg.isIndexStale()).toBe(false);
- await cg.indexAll();
- const info = cg.getIndexBuildInfo();
- expect(info.extractionVersion).toBe(EXTRACTION_VERSION);
- expect(typeof info.version).toBe('string');
- expect(cg.isIndexStale()).toBe(false);
- cg.destroy();
- });
- it('flags an index stamped by an older extraction version as stale', async () => {
- fs.writeFileSync(path.join(dir, 'a.ts'), 'export function hello() { return 1; }\n');
- const cg = await CodeGraph.init(dir, { index: false });
- await cg.indexAll();
- // Simulate an index built by an older engine.
- (cg as unknown as { queries: { setMetadata(k: string, v: string): void } }).queries.setMetadata(
- 'indexed_with_extraction_version',
- String(EXTRACTION_VERSION - 1)
- );
- expect(cg.isIndexStale()).toBe(true);
- cg.destroy();
- });
- });
|