mirror of
https://github.com/alkimake/paperclip.git
synced 2026-06-14 01:50:39 +09:00
## Thinking Path > - Paperclip orchestrates AI agents for zero-human companies > - That operator experience depends not just on issue chat, but also on how workspaces, inbox groups, and navigation state behave over long-running sessions > - The current branch included a separate cluster of workspace-runtime controls, inbox grouping, sidebar ordering, and worktree lifecycle fixes > - Those changes cross server, shared contracts, database state, and UI navigation, but they still form one coherent operator workflow area > - This pull request isolates the workspace/runtime and navigation ergonomics work into one standalone branch > - The benefit is better workspace recovery and navigation persistence without forcing reviewers through the unrelated issue-detail/chat work ## What Changed - Improved execution workspace and project workspace controls, request wiring, layout, and JSON editor ergonomics - Hardened linked worktree reuse/startup behavior and documented the `worktree repair` flow for recovering linked worktrees safely - Added inbox workspace grouping, mobile collapse, archive undo, keyboard navigation, shared group-header styling, and persisted collapsed-group behavior - Added persistent sidebar order preferences with the supporting DB migration, shared/server contracts, routes, services, hooks, and UI integration - Scoped issue-list preferences by context and added targeted UI/server tests for workspace controls, inbox behavior, sidebar preferences, and worktree validation ## Verification - `pnpm vitest run server/src/__tests__/sidebar-preferences-routes.test.ts ui/src/pages/Inbox.test.tsx ui/src/components/ProjectWorkspaceSummaryCard.test.tsx ui/src/components/WorkspaceRuntimeControls.test.tsx ui/src/api/workspace-runtime-control.test.ts` - `server/src/__tests__/workspace-runtime.test.ts` was attempted, but the embedded Postgres suite self-skipped/hung on this host after reporting an init-script issue, so it is not counted as a local pass here ## Risks - Medium: this branch includes migration-backed preference storage plus worktree/runtime behavior, so merge review should pay attention to state persistence and worktree recovery semantics - The sidebar preference migration is standalone, but it should still be watched for conflicts if another migration lands first ## Model Used - OpenAI Codex coding agent (GPT-5-class runtime in Codex CLI; exact deployed model ID is not exposed in this environment), reasoning enabled, tool use and local code execution enabled ## Checklist - [x] I have included a thinking path that traces from project context to this change - [x] I have specified the model used (with version and capability details) - [ ] I have run tests locally and they pass - [x] I have added or updated tests where applicable - [ ] If this change affects the UI, I have included before/after screenshots - [x] I have updated relevant documentation to reflect my changes - [x] I have considered and documented any risks above - [x] I will address all Greptile and reviewer comments before requesting merge --------- Co-authored-by: Paperclip <noreply@paperclip.ing>
131 lines
4.7 KiB
TypeScript
131 lines
4.7 KiB
TypeScript
import { z } from "zod";
|
|
|
|
export const executionWorkspaceStatusSchema = z.enum([
|
|
"active",
|
|
"idle",
|
|
"in_review",
|
|
"archived",
|
|
"cleanup_failed",
|
|
]);
|
|
|
|
export const executionWorkspaceConfigSchema = z.object({
|
|
provisionCommand: z.string().optional().nullable(),
|
|
teardownCommand: z.string().optional().nullable(),
|
|
cleanupCommand: z.string().optional().nullable(),
|
|
workspaceRuntime: z.record(z.unknown()).optional().nullable(),
|
|
desiredState: z.enum(["running", "stopped"]).optional().nullable(),
|
|
serviceStates: z.record(z.enum(["running", "stopped"])).optional().nullable(),
|
|
}).strict();
|
|
|
|
export const workspaceRuntimeControlTargetSchema = z.object({
|
|
workspaceCommandId: z.string().min(1).optional().nullable(),
|
|
runtimeServiceId: z.string().uuid().optional().nullable(),
|
|
serviceIndex: z.number().int().nonnegative().optional().nullable(),
|
|
}).strict();
|
|
|
|
export const executionWorkspaceCloseReadinessStateSchema = z.enum([
|
|
"ready",
|
|
"ready_with_warnings",
|
|
"blocked",
|
|
]);
|
|
|
|
export const executionWorkspaceCloseActionKindSchema = z.enum([
|
|
"archive_record",
|
|
"stop_runtime_services",
|
|
"cleanup_command",
|
|
"teardown_command",
|
|
"git_worktree_remove",
|
|
"git_branch_delete",
|
|
"remove_local_directory",
|
|
]);
|
|
|
|
export const executionWorkspaceCloseActionSchema = z.object({
|
|
kind: executionWorkspaceCloseActionKindSchema,
|
|
label: z.string(),
|
|
description: z.string(),
|
|
command: z.string().nullable(),
|
|
}).strict();
|
|
|
|
export const executionWorkspaceCloseLinkedIssueSchema = z.object({
|
|
id: z.string().uuid(),
|
|
identifier: z.string().nullable(),
|
|
title: z.string(),
|
|
status: z.string(),
|
|
isTerminal: z.boolean(),
|
|
}).strict();
|
|
|
|
export const executionWorkspaceCloseGitReadinessSchema = z.object({
|
|
repoRoot: z.string().nullable(),
|
|
workspacePath: z.string().nullable(),
|
|
branchName: z.string().nullable(),
|
|
baseRef: z.string().nullable(),
|
|
hasDirtyTrackedFiles: z.boolean(),
|
|
hasUntrackedFiles: z.boolean(),
|
|
dirtyEntryCount: z.number().int().nonnegative(),
|
|
untrackedEntryCount: z.number().int().nonnegative(),
|
|
aheadCount: z.number().int().nonnegative().nullable(),
|
|
behindCount: z.number().int().nonnegative().nullable(),
|
|
isMergedIntoBase: z.boolean().nullable(),
|
|
createdByRuntime: z.boolean(),
|
|
}).strict();
|
|
|
|
export const workspaceRuntimeServiceSchema = z.object({
|
|
id: z.string(),
|
|
companyId: z.string().uuid(),
|
|
projectId: z.string().uuid().nullable(),
|
|
projectWorkspaceId: z.string().uuid().nullable(),
|
|
executionWorkspaceId: z.string().uuid().nullable(),
|
|
issueId: z.string().uuid().nullable(),
|
|
scopeType: z.enum(["project_workspace", "execution_workspace", "run", "agent"]),
|
|
scopeId: z.string().nullable(),
|
|
serviceName: z.string(),
|
|
status: z.enum(["starting", "running", "stopped", "failed"]),
|
|
lifecycle: z.enum(["shared", "ephemeral"]),
|
|
reuseKey: z.string().nullable(),
|
|
command: z.string().nullable(),
|
|
cwd: z.string().nullable(),
|
|
port: z.number().int().nullable(),
|
|
url: z.string().nullable(),
|
|
provider: z.enum(["local_process", "adapter_managed"]),
|
|
providerRef: z.string().nullable(),
|
|
ownerAgentId: z.string().uuid().nullable(),
|
|
startedByRunId: z.string().uuid().nullable(),
|
|
lastUsedAt: z.coerce.date(),
|
|
startedAt: z.coerce.date(),
|
|
stoppedAt: z.coerce.date().nullable(),
|
|
stopPolicy: z.record(z.unknown()).nullable(),
|
|
healthStatus: z.enum(["unknown", "healthy", "unhealthy"]),
|
|
configIndex: z.number().int().nonnegative().nullable().optional(),
|
|
createdAt: z.coerce.date(),
|
|
updatedAt: z.coerce.date(),
|
|
}).strict();
|
|
|
|
export const executionWorkspaceCloseReadinessSchema = z.object({
|
|
workspaceId: z.string().uuid(),
|
|
state: executionWorkspaceCloseReadinessStateSchema,
|
|
blockingReasons: z.array(z.string()),
|
|
warnings: z.array(z.string()),
|
|
linkedIssues: z.array(executionWorkspaceCloseLinkedIssueSchema),
|
|
plannedActions: z.array(executionWorkspaceCloseActionSchema),
|
|
isDestructiveCloseAllowed: z.boolean(),
|
|
isSharedWorkspace: z.boolean(),
|
|
isProjectPrimaryWorkspace: z.boolean(),
|
|
git: executionWorkspaceCloseGitReadinessSchema.nullable(),
|
|
runtimeServices: z.array(workspaceRuntimeServiceSchema),
|
|
}).strict();
|
|
|
|
export const updateExecutionWorkspaceSchema = z.object({
|
|
name: z.string().min(1).optional(),
|
|
cwd: z.string().optional().nullable(),
|
|
repoUrl: z.string().optional().nullable(),
|
|
baseRef: z.string().optional().nullable(),
|
|
branchName: z.string().optional().nullable(),
|
|
providerRef: z.string().optional().nullable(),
|
|
status: executionWorkspaceStatusSchema.optional(),
|
|
cleanupEligibleAt: z.string().datetime().optional().nullable(),
|
|
cleanupReason: z.string().optional().nullable(),
|
|
config: executionWorkspaceConfigSchema.optional().nullable(),
|
|
metadata: z.record(z.unknown()).optional().nullable(),
|
|
}).strict();
|
|
|
|
export type UpdateExecutionWorkspace = z.infer<typeof updateExecutionWorkspaceSchema>;
|