test-workspace-path-expansion.js 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. #!/usr/bin/env node
  2. import assert from 'assert';
  3. import express from 'express';
  4. import fs from 'fs';
  5. import os from 'os';
  6. import path from 'path';
  7. import { setupFileRoutes } from './src/server/routes/files.js';
  8. import { setupWorkspaceRoutes } from './src/server/routes/workspace.js';
  9. import { VL_CODE_HOME, loadInstances } from './src/server/helpers.js';
  10. const WORKSPACES_FILE = path.join(VL_CODE_HOME, 'workspaces.json');
  11. const INSTANCES_FILE = path.join(VL_CODE_HOME, 'instances.json');
  12. async function main() {
  13. const createParent = fs.mkdtempSync(path.join(os.homedir(), 'vlcode-home-create-'));
  14. const openDir = fs.mkdtempSync(path.join(os.tmpdir(), 'vlcode-open-workspace-'));
  15. const workspacesBackup = fs.existsSync(WORKSPACES_FILE)
  16. ? fs.readFileSync(WORKSPACES_FILE, 'utf8')
  17. : null;
  18. const instancesBackup = fs.existsSync(INSTANCES_FILE)
  19. ? fs.readFileSync(INSTANCES_FILE, 'utf8')
  20. : null;
  21. const app = express();
  22. app.use(express.json({ limit: '10mb' }));
  23. const fakeServer = {
  24. config: {
  25. emptyWorkspaceDir: path.join(os.tmpdir(), 'vlcode-empty-window-route-test'),
  26. workDir: openDir,
  27. },
  28. _serverPort: 4002,
  29. _openTabs: [],
  30. _workspaceStates: new Map(),
  31. _saveOpenTabs() {},
  32. broadcast() {},
  33. hasSelectedWorkspace() { return false; },
  34. getVisibleWorkDir() { return ''; },
  35. async switchWorkspace(dirPath) { return { ok: true, workDir: dirPath }; },
  36. };
  37. setupFileRoutes(app, fakeServer);
  38. setupWorkspaceRoutes(app, fakeServer);
  39. const httpServer = app.listen(0);
  40. await new Promise((resolve, reject) => {
  41. httpServer.once('listening', resolve);
  42. httpServer.once('error', reject);
  43. });
  44. const { port } = httpServer.address();
  45. let openedPort = null;
  46. try {
  47. const browseRes = await fetch(`http://127.0.0.1:${port}/api/browse-dir?path=${encodeURIComponent('~')}`);
  48. const browseBody = await browseRes.json();
  49. assert.equal(browseRes.status, 200);
  50. assert.equal(browseBody.current, os.homedir(), 'tilde path should resolve to the real home directory');
  51. const createParentTilde = `~/${path.basename(createParent)}`;
  52. const createRes = await fetch(`http://127.0.0.1:${port}/api/workspaces/create-project`, {
  53. method: 'POST',
  54. headers: { 'Content-Type': 'application/json' },
  55. body: JSON.stringify({
  56. projectName: 'TildeCreateRouteTest',
  57. parentDir: createParentTilde,
  58. }),
  59. });
  60. const createBody = await createRes.json();
  61. assert.equal(createRes.status, 200);
  62. assert.equal(createBody.path, path.join(createParent, 'TildeCreateRouteTest'));
  63. assert.ok(fs.existsSync(path.join(createBody.path, '.vl-code', 'project.json')));
  64. const openRes = await fetch(`http://127.0.0.1:${port}/api/windows/open`, {
  65. method: 'POST',
  66. headers: { 'Content-Type': 'application/json' },
  67. body: JSON.stringify({ dirPath: openDir }),
  68. });
  69. const openBody = await openRes.json();
  70. assert.equal(openRes.status, 200, openBody.error || 'open route should succeed');
  71. assert.equal(typeof openBody.port, 'number');
  72. openedPort = openBody.port;
  73. const healthRes = await fetch(`http://127.0.0.1:${openedPort}/api/health`);
  74. const healthBody = await healthRes.json();
  75. assert.equal(healthRes.status, 200);
  76. assert.equal(healthBody.ok, true);
  77. const closeRes = await fetch(`http://127.0.0.1:${port}/api/windows/${openedPort}`, { method: 'DELETE' });
  78. const closeBody = await closeRes.json();
  79. assert.equal(closeRes.status, 200);
  80. assert.equal(closeBody.ok, true);
  81. openedPort = null;
  82. } finally {
  83. if (openedPort) {
  84. const live = loadInstances().find((entry) => entry.port === openedPort);
  85. if (live) {
  86. try { process.kill(live.pid, 'SIGTERM'); } catch {}
  87. }
  88. }
  89. await new Promise((resolve) => httpServer.close(resolve));
  90. fs.rmSync(createParent, { recursive: true, force: true });
  91. fs.rmSync(openDir, { recursive: true, force: true });
  92. if (workspacesBackup === null) fs.rmSync(WORKSPACES_FILE, { force: true });
  93. else fs.writeFileSync(WORKSPACES_FILE, workspacesBackup, 'utf8');
  94. if (instancesBackup === null) fs.rmSync(INSTANCES_FILE, { force: true });
  95. else fs.writeFileSync(INSTANCES_FILE, instancesBackup, 'utf8');
  96. }
  97. console.log('\n── Workspace Path Expansion ──');
  98. console.log('PASS test-workspace-path-expansion.js');
  99. }
  100. main().catch((err) => {
  101. console.error(err);
  102. process.exit(1);
  103. });