mirror of
https://github.com/siteboon/claudecodeui.git
synced 2026-05-01 10:18:37 +00:00
Restructure project creation, listing, GitHub clone progress, and TaskMaster details behind a dedicated TypeScript module under server/modules/projects/, and align the client wizard with a single path-based flow. Server / routing - Remove server/routes/projects.js and mount server/modules/projects/ projects.routes.ts at /api/projects (still behind authenticateToken). - Drop duplicate handlers from server/index.js for GET /api/projects and GET /api/projects/:projectId/taskmaster; those live on the new router. - Import WORKSPACES_ROOT and validateWorkspacePath from shared utils in index.js instead of the deleted projects route module. Projects router (projects.routes.ts) - GET /: list projects with sessions (existing snapshot behavior). - POST /create-project: validate body, reject legacy workspaceType and mixed clone fields, delegate to createProject service, return distinct success copy when an archived path is reactivated. - GET /clone-progress: Server-Sent Events for clone progress/complete/error; requires authenticated user id for token resolution; wires startCloneProject. - GET /:projectId/taskmaster: delegates to getProjectTaskMaster. Services (new) - project-management.service.ts: path validation, workspace directory creation, persistence via projectsDb.createProjectPath, mapping to API project shape; surfaces AppError for validation, conflict, and not-found cases; optional dependency injection for tests. - project-clone.service.ts: validates workspace, resolves GitHub auth (stored token or inline token), runs git clone with progress callbacks, registers project via createProject on success; sanitizes errors and supports cancellation; injectable dependencies for tests. - projects-has-taskmaster.service.ts: moves TaskMaster detection and normalization out of server/projects.js; resolve-by-id and public getProjectTaskMaster with structured AppError responses. Persistence and shared types - projectsDb.createProjectPath now returns CreateProjectPathResult (created | reactivated_archived | active_conflict) using INSERT … ON CONFLICT with selective update when the row is archived; normalizes display name from path or custom name; repository row typing moves to shared ProjectRepositoryRow. - getProjectPaths() returns only non-archived rows (isArchived = 0). - shared/types.ts: ProjectRepositoryRow, CreateProjectPathResult/outcome, WorkspacePathValidationResult. - shared/utils.ts: WORKSPACES_ROOT, forbidden path lists, validateWorkspacePath, asyncHandler for Express async routes. Legacy cleanup - server/projects.js: remove detectTaskMasterFolder, normalizeTaskMasterInfo, and getProjectTaskMasterById (logic lives in the new service). - server/routes/agent.js: register external API project paths with projectsDb.createProjectPath instead of addProjectManually try/catch; treat active_conflict as an existing registration and continue. Tests - Add Node test suites for project-management, project-clone, and projects-has-taskmaster services; update projects.service test import for renamed projects-with-sessions-fetch.service.ts. Rename - projects.service.ts → projects-with-sessions-fetch.service.ts; re-export from modules/projects/index.ts. Client (project creation wizard) - Remove StepTypeSelection and workspaceType from form state and types; wizard is two steps (configure path/GitHub auth, then review). - createWorkspaceRequest → createProjectRequest; clone vs create-only inferred from githubUrl (pathUtils / isCloneWorkflow). - Adjust step indices, WizardProgress, StepConfiguration/Review, WorkspacePathField, and src/utils/api.js as needed for the new API. Docs - Minor websocket README touch-up. Net: ~1.6k insertions / ~0.9k deletions across 29 files; behavior is centralized in typed services with explicit HTTP errors and test seams.
118 lines
3.7 KiB
TypeScript
118 lines
3.7 KiB
TypeScript
import assert from 'node:assert/strict';
|
|
import test from 'node:test';
|
|
|
|
import { createProject } from '@/modules/projects/services/project-management.service.js';
|
|
import { AppError } from '@/shared/utils.js';
|
|
|
|
const projectRow = {
|
|
project_id: 'project-1',
|
|
project_path: '/workspace/my-project',
|
|
custom_project_name: 'my-project',
|
|
isStarred: 0,
|
|
isArchived: 0,
|
|
};
|
|
|
|
test('createProject throws when project path is missing', async () => {
|
|
await assert.rejects(
|
|
async () => createProject({ projectPath: '' }),
|
|
(error: unknown) => {
|
|
assert.ok(error instanceof AppError);
|
|
assert.equal(error.code, 'PROJECT_PATH_REQUIRED');
|
|
assert.equal(error.statusCode, 400);
|
|
return true;
|
|
},
|
|
);
|
|
});
|
|
|
|
test('createProject throws when path validation fails', async () => {
|
|
await assert.rejects(
|
|
async () =>
|
|
createProject(
|
|
{ projectPath: '/invalid/path' },
|
|
{
|
|
validatePath: async () => ({ valid: false, error: 'blocked path' }),
|
|
ensureWorkspaceDirectory: async () => undefined,
|
|
persistProjectPath: () => ({ outcome: 'created', project: projectRow }),
|
|
getProjectByPath: () => projectRow,
|
|
},
|
|
),
|
|
(error: unknown) => {
|
|
assert.ok(error instanceof AppError);
|
|
assert.equal(error.code, 'INVALID_PROJECT_PATH');
|
|
assert.equal(error.statusCode, 400);
|
|
assert.equal(error.details, 'blocked path');
|
|
return true;
|
|
},
|
|
);
|
|
});
|
|
|
|
test('createProject throws conflict when active project path already exists', async () => {
|
|
await assert.rejects(
|
|
async () =>
|
|
createProject(
|
|
{ projectPath: '/workspace/my-project' },
|
|
{
|
|
validatePath: async () => ({ valid: true, resolvedPath: '/workspace/my-project' }),
|
|
ensureWorkspaceDirectory: async () => undefined,
|
|
persistProjectPath: () => ({ outcome: 'active_conflict', project: projectRow }),
|
|
getProjectByPath: () => projectRow,
|
|
},
|
|
),
|
|
(error: unknown) => {
|
|
assert.ok(error instanceof AppError);
|
|
assert.equal(error.code, 'PROJECT_ALREADY_EXISTS');
|
|
assert.equal(error.statusCode, 409);
|
|
assert.equal(error.details, 'Project path already exists: /workspace/my-project');
|
|
return true;
|
|
},
|
|
);
|
|
});
|
|
|
|
test('createProject falls back to directory name when custom name is not provided', async () => {
|
|
let capturedCustomName: string | null = null;
|
|
|
|
const result = await createProject(
|
|
{ projectPath: '/workspace/my-project', customName: '' },
|
|
{
|
|
validatePath: async () => ({ valid: true, resolvedPath: '/workspace/my-project' }),
|
|
ensureWorkspaceDirectory: async () => undefined,
|
|
persistProjectPath: (_projectPath, customName) => {
|
|
capturedCustomName = customName;
|
|
return {
|
|
outcome: 'created',
|
|
project: {
|
|
...projectRow,
|
|
custom_project_name: customName,
|
|
},
|
|
};
|
|
},
|
|
getProjectByPath: () => projectRow,
|
|
},
|
|
);
|
|
|
|
assert.equal(capturedCustomName, 'my-project');
|
|
assert.equal(result.outcome, 'created');
|
|
assert.equal(result.project.displayName, 'my-project');
|
|
});
|
|
|
|
test('createProject returns archived reuse outcome when archived row is reused', async () => {
|
|
const result = await createProject(
|
|
{ projectPath: '/workspace/my-project' },
|
|
{
|
|
validatePath: async () => ({ valid: true, resolvedPath: '/workspace/my-project' }),
|
|
ensureWorkspaceDirectory: async () => undefined,
|
|
persistProjectPath: () => ({
|
|
outcome: 'reactivated_archived',
|
|
project: {
|
|
...projectRow,
|
|
isArchived: 1,
|
|
},
|
|
}),
|
|
getProjectByPath: () => projectRow,
|
|
},
|
|
);
|
|
|
|
assert.equal(result.outcome, 'reactivated_archived');
|
|
assert.equal(result.project.isArchived, true);
|
|
});
|