include-ignored-config.test.ts 3.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889
  1. /**
  2. * `codegraph.json` `includeIgnored` loader (#970, #976 / #622, #699).
  3. *
  4. * Parsing, validation, and mtime-caching of the opt-in patterns that re-include
  5. * gitignored directories for embedded-repo discovery. The behavioral end of this
  6. * feature (scanDirectory / discoverEmbeddedRepoRoots / sync honoring the patterns)
  7. * lives in `multi-repo-workspace.test.ts`; these are the loader unit tests,
  8. * mirroring the `extensions` loader coverage in `extension-mapping.test.ts`.
  9. *
  10. * Invariant under test: every failure mode degrades to the zero-config default
  11. * (empty patterns → `.gitignore` fully respected), never a throw.
  12. */
  13. import { describe, it, expect, beforeEach, afterEach } from 'vitest';
  14. import * as fs from 'node:fs';
  15. import * as path from 'node:path';
  16. import * as os from 'node:os';
  17. import { loadIncludeIgnoredPatterns, loadExtensionOverrides, clearProjectConfigCache } from '../src/project-config';
  18. describe('includeIgnored loader (codegraph.json)', () => {
  19. let dir: string;
  20. beforeEach(() => {
  21. dir = fs.mkdtempSync(path.join(os.tmpdir(), 'cg-includeignored-'));
  22. clearProjectConfigCache();
  23. });
  24. afterEach(() => {
  25. clearProjectConfigCache();
  26. fs.rmSync(dir, { recursive: true, force: true });
  27. });
  28. const writeConfig = (obj: unknown) =>
  29. fs.writeFileSync(
  30. path.join(dir, 'codegraph.json'),
  31. typeof obj === 'string' ? obj : JSON.stringify(obj)
  32. );
  33. it('returns an empty list when there is no codegraph.json (the default)', () => {
  34. expect(loadIncludeIgnoredPatterns(dir)).toEqual([]);
  35. });
  36. it('loads a well-formed pattern array', () => {
  37. writeConfig({ includeIgnored: ['packages/', 'services/'] });
  38. expect(loadIncludeIgnoredPatterns(dir)).toEqual(['packages/', 'services/']);
  39. });
  40. it('trims whitespace and drops blank / non-string entries', () => {
  41. writeConfig({ includeIgnored: [' packages/ ', '', ' ', 42, null, 'services/'] });
  42. expect(loadIncludeIgnoredPatterns(dir)).toEqual(['packages/', 'services/']);
  43. });
  44. it('ignores a non-array includeIgnored value without throwing', () => {
  45. writeConfig({ includeIgnored: 'packages/' });
  46. expect(loadIncludeIgnoredPatterns(dir)).toEqual([]);
  47. });
  48. it('ignores malformed JSON without throwing', () => {
  49. writeConfig('{ not: valid json ');
  50. expect(loadIncludeIgnoredPatterns(dir)).toEqual([]);
  51. });
  52. it('returns [] when the field is absent but other config is present', () => {
  53. writeConfig({ extensions: { '.foo': 'typescript' } });
  54. expect(loadIncludeIgnoredPatterns(dir)).toEqual([]);
  55. });
  56. it('coexists with extensions in one file (shared single parse)', () => {
  57. writeConfig({ extensions: { '.foo': 'typescript' }, includeIgnored: ['vendor/'] });
  58. expect(loadExtensionOverrides(dir)).toEqual({ '.foo': 'typescript' });
  59. expect(loadIncludeIgnoredPatterns(dir)).toEqual(['vendor/']);
  60. });
  61. it('picks up a changed config (mtime-invalidated cache)', () => {
  62. writeConfig({ includeIgnored: ['packages/'] });
  63. expect(loadIncludeIgnoredPatterns(dir)).toEqual(['packages/']);
  64. writeConfig({ includeIgnored: ['services/'] });
  65. // Force a distinct mtime in case the filesystem clock is coarse.
  66. const future = new Date(Date.now() + 2000);
  67. fs.utimesSync(path.join(dir, 'codegraph.json'), future, future);
  68. expect(loadIncludeIgnoredPatterns(dir)).toEqual(['services/']);
  69. });
  70. it('drops the patterns again when the config file is removed', () => {
  71. writeConfig({ includeIgnored: ['packages/'] });
  72. expect(loadIncludeIgnoredPatterns(dir)).toEqual(['packages/']);
  73. fs.rmSync(path.join(dir, 'codegraph.json'));
  74. expect(loadIncludeIgnoredPatterns(dir)).toEqual([]);
  75. });
  76. });