@tom2012/cc-web 1.5.10

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (59) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +339 -0
  3. package/backend/dist/auth.d.ts +15 -0
  4. package/backend/dist/auth.d.ts.map +1 -0
  5. package/backend/dist/auth.js +92 -0
  6. package/backend/dist/auth.js.map +1 -0
  7. package/backend/dist/config.d.ts +33 -0
  8. package/backend/dist/config.d.ts.map +1 -0
  9. package/backend/dist/config.js +155 -0
  10. package/backend/dist/config.js.map +1 -0
  11. package/backend/dist/index.d.ts +3 -0
  12. package/backend/dist/index.d.ts.map +1 -0
  13. package/backend/dist/index.js +499 -0
  14. package/backend/dist/index.js.map +1 -0
  15. package/backend/dist/routes/auth.d.ts +3 -0
  16. package/backend/dist/routes/auth.d.ts.map +1 -0
  17. package/backend/dist/routes/auth.js +108 -0
  18. package/backend/dist/routes/auth.js.map +1 -0
  19. package/backend/dist/routes/filesystem.d.ts +3 -0
  20. package/backend/dist/routes/filesystem.d.ts.map +1 -0
  21. package/backend/dist/routes/filesystem.js +243 -0
  22. package/backend/dist/routes/filesystem.js.map +1 -0
  23. package/backend/dist/routes/projects.d.ts +3 -0
  24. package/backend/dist/routes/projects.d.ts.map +1 -0
  25. package/backend/dist/routes/projects.js +235 -0
  26. package/backend/dist/routes/projects.js.map +1 -0
  27. package/backend/dist/routes/shortcuts.d.ts +3 -0
  28. package/backend/dist/routes/shortcuts.d.ts.map +1 -0
  29. package/backend/dist/routes/shortcuts.js +88 -0
  30. package/backend/dist/routes/shortcuts.js.map +1 -0
  31. package/backend/dist/routes/update.d.ts +3 -0
  32. package/backend/dist/routes/update.d.ts.map +1 -0
  33. package/backend/dist/routes/update.js +104 -0
  34. package/backend/dist/routes/update.js.map +1 -0
  35. package/backend/dist/session-manager.d.ts +47 -0
  36. package/backend/dist/session-manager.d.ts.map +1 -0
  37. package/backend/dist/session-manager.js +345 -0
  38. package/backend/dist/session-manager.js.map +1 -0
  39. package/backend/dist/terminal-manager.d.ts +27 -0
  40. package/backend/dist/terminal-manager.d.ts.map +1 -0
  41. package/backend/dist/terminal-manager.js +211 -0
  42. package/backend/dist/terminal-manager.js.map +1 -0
  43. package/backend/dist/types.d.ts +17 -0
  44. package/backend/dist/types.d.ts.map +1 -0
  45. package/backend/dist/types.js +3 -0
  46. package/backend/dist/types.js.map +1 -0
  47. package/backend/dist/usage-terminal.d.ts +18 -0
  48. package/backend/dist/usage-terminal.d.ts.map +1 -0
  49. package/backend/dist/usage-terminal.js +189 -0
  50. package/backend/dist/usage-terminal.js.map +1 -0
  51. package/backend/package-lock.json +1965 -0
  52. package/backend/package.json +31 -0
  53. package/bin/ccweb.js +478 -0
  54. package/electron/dist/main.js +455 -0
  55. package/frontend/dist/assets/index-CQjbS4zv.css +32 -0
  56. package/frontend/dist/assets/index-CtyR65A4.js +434 -0
  57. package/frontend/dist/index.html +14 -0
  58. package/frontend/dist/terminal.svg +4 -0
  59. package/package.json +88 -0
@@ -0,0 +1,3 @@
1
+ declare const router: import("express-serve-static-core").Router;
2
+ export default router;
3
+ //# sourceMappingURL=filesystem.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"filesystem.d.ts","sourceRoot":"","sources":["../../src/routes/filesystem.ts"],"names":[],"mappings":"AAMA,QAAA,MAAM,MAAM,4CAAW,CAAC;AA+NxB,eAAe,MAAM,CAAC"}
@@ -0,0 +1,243 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ const express_1 = require("express");
37
+ const fs = __importStar(require("fs"));
38
+ const path = __importStar(require("path"));
39
+ const os = __importStar(require("os"));
40
+ const config_1 = require("../config");
41
+ const router = (0, express_1.Router)();
42
+ /**
43
+ * Security: restrict filesystem access to registered project directories and home dir.
44
+ * Prevents path traversal attacks (e.g. reading /etc/shadow).
45
+ * Also resolves symlinks to prevent symlink-based traversal attacks.
46
+ */
47
+ function isWithinAllowedDirs(p) {
48
+ const home = os.homedir();
49
+ if (p === home || p.startsWith(home + path.sep))
50
+ return true;
51
+ const projects = (0, config_1.getProjects)();
52
+ for (const proj of projects) {
53
+ const projectDir = path.resolve(proj.folderPath);
54
+ if (p === projectDir || p.startsWith(projectDir + path.sep))
55
+ return true;
56
+ }
57
+ return false;
58
+ }
59
+ function isPathAllowed(resolvedPath) {
60
+ if (!isWithinAllowedDirs(resolvedPath))
61
+ return false;
62
+ // Also verify the real path (after symlink resolution) is within allowed dirs.
63
+ // This prevents symlink attacks where a link inside an allowed dir points outside.
64
+ try {
65
+ const realPath = fs.realpathSync(resolvedPath);
66
+ if (realPath !== resolvedPath && !isWithinAllowedDirs(realPath))
67
+ return false;
68
+ }
69
+ catch {
70
+ // Path may not exist yet (e.g. for new file writes) — skip realpath check
71
+ }
72
+ return true;
73
+ }
74
+ // GET /api/filesystem?path=...
75
+ router.get('/', (req, res) => {
76
+ const requestedPath = req.query['path'] || os.homedir();
77
+ // Normalize and resolve the path
78
+ const resolvedPath = path.resolve(requestedPath);
79
+ if (!isPathAllowed(resolvedPath)) {
80
+ res.status(403).json({ error: 'Access denied: path outside allowed directories' });
81
+ return;
82
+ }
83
+ let entries = [];
84
+ try {
85
+ const dirents = fs.readdirSync(resolvedPath, { withFileTypes: true });
86
+ entries = dirents
87
+ .map((d) => ({
88
+ name: d.name,
89
+ type: (d.isDirectory() ? 'dir' : 'file'),
90
+ path: path.join(resolvedPath, d.name),
91
+ }))
92
+ .sort((a, b) => {
93
+ // Directories first, then alphabetical
94
+ if (a.type === 'dir' && b.type !== 'dir')
95
+ return -1;
96
+ if (a.type !== 'dir' && b.type === 'dir')
97
+ return 1;
98
+ return a.name.localeCompare(b.name);
99
+ });
100
+ }
101
+ catch (err) {
102
+ const error = err;
103
+ if (error.code === 'EACCES') {
104
+ res.status(403).json({ error: 'Permission denied', path: resolvedPath });
105
+ return;
106
+ }
107
+ if (error.code === 'ENOENT') {
108
+ res.status(404).json({ error: 'Path not found', path: resolvedPath });
109
+ return;
110
+ }
111
+ res.status(500).json({ error: 'Failed to read directory', path: resolvedPath });
112
+ return;
113
+ }
114
+ const parent = path.dirname(resolvedPath);
115
+ res.json({
116
+ path: resolvedPath,
117
+ parent: parent !== resolvedPath ? parent : null,
118
+ entries,
119
+ });
120
+ });
121
+ // POST /api/filesystem/mkdir body: { path: string, name: string }
122
+ router.post('/mkdir', (req, res) => {
123
+ const { path: parentPath, name } = req.body;
124
+ if (!parentPath || !name) {
125
+ res.status(400).json({ error: 'path and name are required' });
126
+ return;
127
+ }
128
+ // Reject names with path separators, parent traversal, or dangerous characters
129
+ if (/[/\\\0:]/.test(name) || name === '..' || name === '.' || name.trim() !== name || name.length > 255) {
130
+ res.status(400).json({ error: 'Invalid folder name' });
131
+ return;
132
+ }
133
+ const resolvedParent = path.resolve(parentPath);
134
+ if (!isPathAllowed(resolvedParent)) {
135
+ res.status(403).json({ error: 'Access denied: path outside allowed directories' });
136
+ return;
137
+ }
138
+ const newDir = path.join(resolvedParent, name);
139
+ try {
140
+ fs.mkdirSync(newDir, { recursive: false });
141
+ res.json({ path: newDir });
142
+ }
143
+ catch (err) {
144
+ const error = err;
145
+ if (error.code === 'EEXIST') {
146
+ res.status(409).json({ error: 'Folder already exists' });
147
+ return;
148
+ }
149
+ if (error.code === 'EACCES') {
150
+ res.status(403).json({ error: 'Permission denied' });
151
+ return;
152
+ }
153
+ res.status(500).json({ error: 'Failed to create folder' });
154
+ }
155
+ });
156
+ // GET /api/filesystem/file?path=...
157
+ router.get('/file', (req, res) => {
158
+ const requestedPath = req.query['path'];
159
+ if (!requestedPath) {
160
+ res.status(400).json({ error: 'path is required' });
161
+ return;
162
+ }
163
+ const resolvedPath = path.resolve(requestedPath);
164
+ if (!isPathAllowed(resolvedPath)) {
165
+ res.status(403).json({ error: 'Access denied: path outside allowed directories' });
166
+ return;
167
+ }
168
+ const SIZE_LIMIT = parseInt(process.env.CCWEB_FILE_SIZE_LIMIT || '', 10) || 1 * 1024 * 1024; // 1 MB default
169
+ let stat;
170
+ try {
171
+ stat = fs.statSync(resolvedPath);
172
+ }
173
+ catch (err) {
174
+ const error = err;
175
+ if (error.code === 'ENOENT') {
176
+ res.status(404).json({ error: 'File not found' });
177
+ return;
178
+ }
179
+ res.status(500).json({ error: 'Failed to stat file' });
180
+ return;
181
+ }
182
+ if (!stat.isFile()) {
183
+ res.status(400).json({ error: 'Path is not a file' });
184
+ return;
185
+ }
186
+ if (stat.size > SIZE_LIMIT) {
187
+ res.json({ path: resolvedPath, tooLarge: true, size: stat.size, content: null });
188
+ return;
189
+ }
190
+ let buffer;
191
+ try {
192
+ buffer = fs.readFileSync(resolvedPath);
193
+ }
194
+ catch (err) {
195
+ const error = err;
196
+ if (error.code === 'EACCES') {
197
+ res.status(403).json({ error: 'Permission denied' });
198
+ return;
199
+ }
200
+ res.status(500).json({ error: 'Failed to read file' });
201
+ return;
202
+ }
203
+ // Heuristic binary detection: look for null bytes in first 8KB
204
+ const sample = buffer.slice(0, 8192);
205
+ const isBinary = sample.includes(0);
206
+ if (isBinary) {
207
+ res.json({ path: resolvedPath, binary: true, size: stat.size, content: null });
208
+ return;
209
+ }
210
+ res.json({ path: resolvedPath, binary: false, tooLarge: false, size: stat.size, content: buffer.toString('utf8') });
211
+ });
212
+ // PUT /api/filesystem/file body: { path: string, content: string }
213
+ router.put('/file', (req, res) => {
214
+ const { path: filePath, content } = req.body;
215
+ if (!filePath || content === undefined) {
216
+ res.status(400).json({ error: 'path and content are required' });
217
+ return;
218
+ }
219
+ const resolvedPath = path.resolve(filePath);
220
+ if (!isPathAllowed(resolvedPath)) {
221
+ res.status(403).json({ error: 'Access denied: path outside allowed directories' });
222
+ return;
223
+ }
224
+ const WRITE_LIMIT = parseInt(process.env.CCWEB_FILE_SIZE_LIMIT || '', 10) || 10 * 1024 * 1024; // 10 MB
225
+ if (Buffer.byteLength(content, 'utf-8') > WRITE_LIMIT) {
226
+ res.status(413).json({ error: 'File too large to write' });
227
+ return;
228
+ }
229
+ try {
230
+ fs.writeFileSync(resolvedPath, content, 'utf-8');
231
+ res.json({ path: resolvedPath, size: Buffer.byteLength(content, 'utf-8') });
232
+ }
233
+ catch (err) {
234
+ const error = err;
235
+ if (error.code === 'EACCES') {
236
+ res.status(403).json({ error: 'Permission denied' });
237
+ return;
238
+ }
239
+ res.status(500).json({ error: 'Failed to write file' });
240
+ }
241
+ });
242
+ exports.default = router;
243
+ //# sourceMappingURL=filesystem.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"filesystem.js","sourceRoot":"","sources":["../../src/routes/filesystem.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ declare const router: import("express-serve-static-core").Router;
2
+ export default router;
3
+ //# sourceMappingURL=projects.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"projects.d.ts","sourceRoot":"","sources":["../../src/routes/projects.ts"],"names":[],"mappings":"AAaA,QAAA,MAAM,MAAM,4CAAW,CAAC;AAkOxB,eAAe,MAAM,CAAC"}
@@ -0,0 +1,235 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ const express_1 = require("express");
37
+ const uuid_1 = require("uuid");
38
+ const fs = __importStar(require("fs"));
39
+ const path = __importStar(require("path"));
40
+ const config_1 = require("../config");
41
+ const terminal_manager_1 = require("../terminal-manager");
42
+ const usage_terminal_1 = require("../usage-terminal");
43
+ const session_manager_1 = require("../session-manager");
44
+ const VALID_CLI_TOOLS = ['claude', 'opencode', 'codex', 'qwen'];
45
+ const router = (0, express_1.Router)();
46
+ /** Initialize .notebook/ directory structure in a project folder */
47
+ function initNotebook(folderPath) {
48
+ try {
49
+ const notebookDir = path.join(folderPath, '.notebook', 'pages');
50
+ if (!fs.existsSync(notebookDir)) {
51
+ fs.mkdirSync(notebookDir, { recursive: true });
52
+ }
53
+ const graphFile = path.join(folderPath, '.notebook', 'graph.yaml');
54
+ if (!fs.existsSync(graphFile)) {
55
+ fs.writeFileSync(graphFile, 'pages: []\nrelations: []\n', 'utf-8');
56
+ }
57
+ }
58
+ catch (err) {
59
+ console.error('[Projects] Failed to init .notebook/:', err);
60
+ }
61
+ }
62
+ // GET /api/projects
63
+ router.get('/', (_req, res) => {
64
+ const projects = (0, config_1.getProjects)();
65
+ res.json(projects);
66
+ });
67
+ // GET /api/projects/activity → { [projectId]: lastActivityAt (epoch ms) }
68
+ router.get('/activity', (_req, res) => {
69
+ res.json(terminal_manager_1.terminalManager.getAllActivity());
70
+ });
71
+ // POST /api/projects
72
+ router.post('/', (req, res) => {
73
+ const { name, folderPath, permissionMode, cliTool } = req.body;
74
+ if (!name || !folderPath || !permissionMode) {
75
+ res.status(400).json({ error: 'name, folderPath, and permissionMode are required' });
76
+ return;
77
+ }
78
+ if (permissionMode !== 'limited' && permissionMode !== 'unlimited') {
79
+ res.status(400).json({ error: 'permissionMode must be "limited" or "unlimited"' });
80
+ return;
81
+ }
82
+ if (cliTool && !VALID_CLI_TOOLS.includes(cliTool)) {
83
+ res.status(400).json({ error: `cliTool must be one of: ${VALID_CLI_TOOLS.join(', ')}` });
84
+ return;
85
+ }
86
+ const project = {
87
+ id: (0, uuid_1.v4)(),
88
+ name,
89
+ folderPath,
90
+ permissionMode,
91
+ cliTool: cliTool ?? 'claude',
92
+ createdAt: new Date().toISOString(),
93
+ status: 'running',
94
+ };
95
+ (0, config_1.saveProject)(project);
96
+ // Write .ccweb/project.json into the project folder
97
+ try {
98
+ (0, config_1.writeProjectConfig)(folderPath, project);
99
+ }
100
+ catch (err) {
101
+ console.error('[Projects] Failed to write .ccweb/project.json:', err);
102
+ }
103
+ initNotebook(folderPath);
104
+ // Start terminal; broadcast is a no-op until a WS client connects
105
+ terminal_manager_1.terminalManager.getOrCreate(project);
106
+ res.status(201).json(project);
107
+ });
108
+ // POST /api/projects/open — open an existing project from its folder (.ccweb/project.json)
109
+ router.post('/open', (req, res) => {
110
+ const { folderPath } = req.body;
111
+ if (!folderPath) {
112
+ res.status(400).json({ error: 'folderPath is required' });
113
+ return;
114
+ }
115
+ if (!fs.existsSync(folderPath)) {
116
+ res.status(400).json({ error: 'Folder does not exist' });
117
+ return;
118
+ }
119
+ const config = (0, config_1.readProjectConfig)(folderPath);
120
+ if (!config) {
121
+ res.status(400).json({ error: 'No .ccweb/project.json found in this folder. This folder has not been used as a CC Web project before.' });
122
+ return;
123
+ }
124
+ // Check if this project is already registered
125
+ const existing = (0, config_1.getProjects)().find((p) => p.folderPath === folderPath);
126
+ if (existing) {
127
+ res.status(409).json({ error: 'This project is already open', project: existing });
128
+ return;
129
+ }
130
+ const project = {
131
+ id: config.id,
132
+ name: config.name,
133
+ folderPath,
134
+ permissionMode: config.permissionMode,
135
+ cliTool: config.cliTool ?? 'claude',
136
+ createdAt: config.createdAt,
137
+ status: 'running',
138
+ };
139
+ (0, config_1.saveProject)(project);
140
+ initNotebook(folderPath);
141
+ // Start terminal
142
+ terminal_manager_1.terminalManager.getOrCreate(project);
143
+ res.status(200).json(project);
144
+ });
145
+ // DELETE /api/projects/:id
146
+ router.delete('/:id', (req, res) => {
147
+ const { id } = req.params;
148
+ const project = (0, config_1.getProject)(id);
149
+ if (!project) {
150
+ res.status(404).json({ error: 'Project not found' });
151
+ return;
152
+ }
153
+ terminal_manager_1.terminalManager.stop(id);
154
+ (0, config_1.deleteProject)(id);
155
+ res.json({ success: true });
156
+ });
157
+ // PATCH /api/projects/:id/stop
158
+ router.patch('/:id/stop', (req, res) => {
159
+ const { id } = req.params;
160
+ const project = (0, config_1.getProject)(id);
161
+ if (!project) {
162
+ res.status(404).json({ error: 'Project not found' });
163
+ return;
164
+ }
165
+ terminal_manager_1.terminalManager.stop(id);
166
+ // terminalManager.stop() already sets status='stopped' and saves
167
+ // Re-read the project to return fresh state
168
+ res.json((0, config_1.getProject)(id) ?? project);
169
+ });
170
+ // PATCH /api/projects/:id/start
171
+ router.patch('/:id/start', (req, res) => {
172
+ const { id } = req.params;
173
+ const project = (0, config_1.getProject)(id);
174
+ if (!project) {
175
+ res.status(404).json({ error: 'Project not found' });
176
+ return;
177
+ }
178
+ project.status = 'running';
179
+ (0, config_1.saveProject)(project);
180
+ terminal_manager_1.terminalManager.getOrCreate(project);
181
+ res.json(project);
182
+ });
183
+ // PATCH /api/projects/:id/archive
184
+ router.patch('/:id/archive', (req, res) => {
185
+ const { id } = req.params;
186
+ const project = (0, config_1.getProject)(id);
187
+ if (!project) {
188
+ res.status(404).json({ error: 'Project not found' });
189
+ return;
190
+ }
191
+ // Stop terminal before archiving
192
+ terminal_manager_1.terminalManager.stop(id);
193
+ project.archived = true;
194
+ project.status = 'stopped';
195
+ (0, config_1.saveProject)(project);
196
+ res.json((0, config_1.getProject)(id) ?? project);
197
+ });
198
+ // PATCH /api/projects/:id/unarchive
199
+ router.patch('/:id/unarchive', (req, res) => {
200
+ const { id } = req.params;
201
+ const project = (0, config_1.getProject)(id);
202
+ if (!project) {
203
+ res.status(404).json({ error: 'Project not found' });
204
+ return;
205
+ }
206
+ project.archived = false;
207
+ (0, config_1.saveProject)(project);
208
+ res.json((0, config_1.getProject)(id) ?? project);
209
+ });
210
+ // GET /api/projects/:id/sessions
211
+ router.get('/:id/sessions', (req, res) => {
212
+ const { id } = req.params;
213
+ res.json(session_manager_1.sessionManager.listSessions(id));
214
+ });
215
+ // GET /api/projects/:id/sessions/:sessionId
216
+ router.get('/:id/sessions/:sessionId', (req, res) => {
217
+ const { id, sessionId } = req.params;
218
+ const session = session_manager_1.sessionManager.getSession(id, sessionId);
219
+ if (!session) {
220
+ res.status(404).json({ error: 'Session not found' });
221
+ return;
222
+ }
223
+ res.json(session);
224
+ });
225
+ // GET /api/projects/usage → Claude Code usage via OAuth API
226
+ // Pass ?refresh=true to bust the cache (e.g. after plan upgrade)
227
+ router.get('/usage', (req, res) => {
228
+ if (req.query['refresh'] === 'true')
229
+ usage_terminal_1.usageTerminal.clearUsageCache();
230
+ usage_terminal_1.usageTerminal.queryUsage()
231
+ .then((data) => res.json(data))
232
+ .catch(() => res.json(null));
233
+ });
234
+ exports.default = router;
235
+ //# sourceMappingURL=projects.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"projects.js","sourceRoot":"","sources":["../../src/routes/projects.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ declare const router: import("express-serve-static-core").Router;
2
+ export default router;
3
+ //# sourceMappingURL=shortcuts.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"shortcuts.d.ts","sourceRoot":"","sources":["../../src/routes/shortcuts.ts"],"names":[],"mappings":"AAKA,QAAA,MAAM,MAAM,4CAAW,CAAC;AA2ExB,eAAe,MAAM,CAAC"}
@@ -0,0 +1,88 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const express_1 = require("express");
4
+ const uuid_1 = require("uuid");
5
+ const config_1 = require("../config");
6
+ const router = (0, express_1.Router)();
7
+ /** Detect circular parentId references (A→B→A). Returns true if adding parentId to targetId would create a cycle. */
8
+ function hasCycle(shortcuts, targetId, parentId) {
9
+ let current = parentId;
10
+ const visited = new Set();
11
+ while (current) {
12
+ if (current === targetId || visited.has(current))
13
+ return true;
14
+ visited.add(current);
15
+ current = shortcuts.find((s) => s.id === current)?.parentId;
16
+ }
17
+ return false;
18
+ }
19
+ // GET /api/shortcuts
20
+ router.get('/', (_req, res) => {
21
+ res.json((0, config_1.getGlobalShortcuts)());
22
+ });
23
+ // POST /api/shortcuts
24
+ router.post('/', (req, res) => {
25
+ const { label, command, parentId } = req.body;
26
+ if (!command?.trim()) {
27
+ res.status(400).json({ error: 'command is required' });
28
+ return;
29
+ }
30
+ const shortcuts = (0, config_1.getGlobalShortcuts)();
31
+ if (parentId && !shortcuts.some((s) => s.id === parentId)) {
32
+ res.status(400).json({ error: 'Parent shortcut not found' });
33
+ return;
34
+ }
35
+ const shortcut = {
36
+ id: (0, uuid_1.v4)(),
37
+ label: label?.trim() || command.trim(),
38
+ command: command.trim(),
39
+ ...(parentId ? { parentId } : {}),
40
+ };
41
+ shortcuts.push(shortcut);
42
+ (0, config_1.saveGlobalShortcuts)(shortcuts);
43
+ res.status(201).json(shortcut);
44
+ });
45
+ // PUT /api/shortcuts/:id
46
+ router.put('/:id', (req, res) => {
47
+ const { id } = req.params;
48
+ const { label, command } = req.body;
49
+ if (!command?.trim()) {
50
+ res.status(400).json({ error: 'command is required' });
51
+ return;
52
+ }
53
+ const shortcuts = (0, config_1.getGlobalShortcuts)();
54
+ const idx = shortcuts.findIndex((s) => s.id === id);
55
+ if (idx < 0) {
56
+ res.status(404).json({ error: 'Not found' });
57
+ return;
58
+ }
59
+ const { parentId } = req.body;
60
+ if (parentId && !shortcuts.some((s) => s.id === parentId)) {
61
+ res.status(400).json({ error: 'Parent shortcut not found' });
62
+ return;
63
+ }
64
+ if (parentId && hasCycle(shortcuts, id, parentId)) {
65
+ res.status(400).json({ error: 'Circular parent reference detected' });
66
+ return;
67
+ }
68
+ shortcuts[idx] = {
69
+ id, label: label?.trim() || command.trim(), command: command.trim(),
70
+ ...(parentId ? { parentId } : parentId === null ? {} : { parentId: shortcuts[idx].parentId }),
71
+ };
72
+ (0, config_1.saveGlobalShortcuts)(shortcuts);
73
+ res.json(shortcuts[idx]);
74
+ });
75
+ // DELETE /api/shortcuts/:id
76
+ router.delete('/:id', (req, res) => {
77
+ const { id } = req.params;
78
+ const shortcuts = (0, config_1.getGlobalShortcuts)();
79
+ const filtered = shortcuts.filter((s) => s.id !== id);
80
+ if (filtered.length === shortcuts.length) {
81
+ res.status(404).json({ error: 'Not found' });
82
+ return;
83
+ }
84
+ (0, config_1.saveGlobalShortcuts)(filtered);
85
+ res.json({ success: true });
86
+ });
87
+ exports.default = router;
88
+ //# sourceMappingURL=shortcuts.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"shortcuts.js","sourceRoot":"","sources":["../../src/routes/shortcuts.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ declare const router: import("express-serve-static-core").Router;
2
+ export default router;
3
+ //# sourceMappingURL=update.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"update.d.ts","sourceRoot":"","sources":["../../src/routes/update.ts"],"names":[],"mappings":"AAKA,QAAA,MAAM,MAAM,4CAAW,CAAC;AA2HxB,eAAe,MAAM,CAAC"}