1
0

cli-node-command.test.ts 3.0 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980
  1. /**
  2. * `codegraph node` argument handling (#1044).
  3. *
  4. * File-read mode (`codegraph node -f <file>`) carries no symbol name, but the
  5. * command was defined with a REQUIRED `<name>` positional, so commander.js
  6. * rejected the call with "missing required argument 'name'" before the action
  7. * ever ran — making file mode unreachable from the CLI. `name` is now optional
  8. * (`[name]`); the action validates that a symbol OR a file is supplied.
  9. *
  10. * Exercised end-to-end against the built binary.
  11. */
  12. import { describe, it, expect, beforeEach, afterEach } from 'vitest';
  13. import { execFileSync } from 'child_process';
  14. import * as fs from 'fs';
  15. import * as os from 'os';
  16. import * as path from 'path';
  17. import { CodeGraph } from '../src';
  18. const BIN = path.resolve(__dirname, '../dist/bin/codegraph.js');
  19. function runNode(cwd: string, extraArgs: string[]): { stdout: string; stderr: string; code: number } {
  20. try {
  21. const stdout = execFileSync(process.execPath, [BIN, 'node', ...extraArgs, '-p', cwd], {
  22. encoding: 'utf-8',
  23. env: { ...process.env, CODEGRAPH_NO_DAEMON: '1', CODEGRAPH_WASM_RELAUNCHED: '1' },
  24. stdio: ['ignore', 'pipe', 'pipe'],
  25. });
  26. return { stdout, stderr: '', code: 0 };
  27. } catch (err: any) {
  28. return { stdout: err.stdout ?? '', stderr: err.stderr ?? '', code: err.status ?? 1 };
  29. }
  30. }
  31. describe('codegraph node — argument handling (#1044)', () => {
  32. let tempDir: string;
  33. beforeEach(async () => {
  34. tempDir = fs.mkdtempSync(path.join(os.tmpdir(), 'codegraph-node-cmd-'));
  35. fs.mkdirSync(path.join(tempDir, 'src'));
  36. fs.writeFileSync(path.join(tempDir, 'src/util.ts'), 'export function util(x: number){ return x + 1; }\n');
  37. const cg = CodeGraph.initSync(tempDir);
  38. await cg.indexAll();
  39. cg.close();
  40. });
  41. afterEach(() => {
  42. fs.rmSync(tempDir, { recursive: true, force: true });
  43. });
  44. it('file mode via -f reads the file (was rejected as "missing required argument")', () => {
  45. const { stdout, code } = runNode(tempDir, ['-f', 'src/util.ts']);
  46. expect(code).toBe(0);
  47. expect(stdout).toContain('src/util.ts');
  48. expect(stdout).toContain('export function util');
  49. // The line-numbered Read-parity shape.
  50. expect(stdout).toMatch(/1\s+export function util/);
  51. });
  52. it('a path-like positional still routes to file mode', () => {
  53. const { stdout, code } = runNode(tempDir, ['src/util.ts']);
  54. expect(code).toBe(0);
  55. expect(stdout).toContain('src/util.ts');
  56. expect(stdout).toContain('export function util');
  57. });
  58. it('a bare symbol positional still routes to symbol mode', () => {
  59. const { stdout, code } = runNode(tempDir, ['util']);
  60. expect(code).toBe(0);
  61. expect(stdout).toContain('util');
  62. expect(stdout).toContain('Location:');
  63. });
  64. it('neither symbol nor file gives a usage error, not commander\'s cryptic one', () => {
  65. const { stderr, code } = runNode(tempDir, []);
  66. expect(code).not.toBe(0);
  67. expect(stderr).toMatch(/symbol name|file/i);
  68. expect(stderr).not.toMatch(/missing required argument/);
  69. });
  70. });