Dench workspace: file manager, relation resolution, and chat refactor
File Manager & Filesystem Operations:
- Add FileManagerTree component with drag-and-drop (dnd-kit), inline
rename, right-click context menu, and compact sidebar mode
- Add context-menu component (open, new file/folder, rename, duplicate,
copy, paste, move, delete) rendered via portal
- Add InlineRename component with validation and shake-on-error animation
- Add useWorkspaceWatcher hook with SSE live-reload and polling fallback
- Add API routes: mkdir, rename, copy, move, watch (SSE file-change
events), and DELETE on /api/workspace/file with system-file protection
- Add safeResolveNewPath and isSystemFile helpers to workspace lib
- Replace inline WorkspaceTreeNode in sidebar with shared FileManagerTree
(compact mode), add workspace refresh callback
Object Relation Resolution:
- Resolve relation fields to human-readable display labels server-side
(resolveRelationLabels, resolveDisplayField helpers)
- Add reverse relation discovery (findReverseRelations) — surfaces
incoming links from other objects
- Add display_field column migration (idempotent ALTER TABLE) and
PATCH /api/workspace/objects/[name]/display-field endpoint
- Enrich object API response with relationLabels, reverseRelations,
effectiveDisplayField, and related_object_name per field
- Add RelationCell, RelationChip, ReverseRelationCell, LinkIcon
components to object-table with clickable cross-object navigation
- Add relation label rendering to kanban cards
- Extract ObjectView component in workspace page with display-field
selector dropdown and relation/reverse-relation badge counts
Chat Panel Extraction:
- Extract chat logic from page.tsx into standalone ChatPanel component
with forwardRef/useImperativeHandle for session control
- ChatPanel supports file-scoped sessions (filePath param) and
context-aware file chat sidebar
- Simplify page.tsx to thin orchestrator delegating to ChatPanel
- Add filePath filter to GET /api/web-sessions for scoped session lists
Dependencies:
- Add @dnd-kit/core, @dnd-kit/sortable, @dnd-kit/utilities
- Add duckdbExec and parseRelationValue to workspace lib
Co-authored-by: Cursor <cursoragent@cursor.com>
2026-02-11 19:22:53 -08:00
|
|
|
import { renameSync, existsSync, statSync } from "node:fs";
|
|
|
|
|
import { join, basename } from "node:path";
|
2026-02-16 00:30:13 -08:00
|
|
|
import { safeResolvePath, isSystemFile } from "@/lib/workspace";
|
Dench workspace: file manager, relation resolution, and chat refactor
File Manager & Filesystem Operations:
- Add FileManagerTree component with drag-and-drop (dnd-kit), inline
rename, right-click context menu, and compact sidebar mode
- Add context-menu component (open, new file/folder, rename, duplicate,
copy, paste, move, delete) rendered via portal
- Add InlineRename component with validation and shake-on-error animation
- Add useWorkspaceWatcher hook with SSE live-reload and polling fallback
- Add API routes: mkdir, rename, copy, move, watch (SSE file-change
events), and DELETE on /api/workspace/file with system-file protection
- Add safeResolveNewPath and isSystemFile helpers to workspace lib
- Replace inline WorkspaceTreeNode in sidebar with shared FileManagerTree
(compact mode), add workspace refresh callback
Object Relation Resolution:
- Resolve relation fields to human-readable display labels server-side
(resolveRelationLabels, resolveDisplayField helpers)
- Add reverse relation discovery (findReverseRelations) — surfaces
incoming links from other objects
- Add display_field column migration (idempotent ALTER TABLE) and
PATCH /api/workspace/objects/[name]/display-field endpoint
- Enrich object API response with relationLabels, reverseRelations,
effectiveDisplayField, and related_object_name per field
- Add RelationCell, RelationChip, ReverseRelationCell, LinkIcon
components to object-table with clickable cross-object navigation
- Add relation label rendering to kanban cards
- Extract ObjectView component in workspace page with display-field
selector dropdown and relation/reverse-relation badge counts
Chat Panel Extraction:
- Extract chat logic from page.tsx into standalone ChatPanel component
with forwardRef/useImperativeHandle for session control
- ChatPanel supports file-scoped sessions (filePath param) and
context-aware file chat sidebar
- Simplify page.tsx to thin orchestrator delegating to ChatPanel
- Add filePath filter to GET /api/web-sessions for scoped session lists
Dependencies:
- Add @dnd-kit/core, @dnd-kit/sortable, @dnd-kit/utilities
- Add duckdbExec and parseRelationValue to workspace lib
Co-authored-by: Cursor <cursoragent@cursor.com>
2026-02-11 19:22:53 -08:00
|
|
|
|
|
|
|
|
export const dynamic = "force-dynamic";
|
|
|
|
|
export const runtime = "nodejs";
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* POST /api/workspace/move
|
|
|
|
|
* Body: { sourcePath: string, destinationDir: string }
|
|
|
|
|
*
|
|
|
|
|
* Moves a file or folder into a different directory.
|
|
|
|
|
* System files are protected from moving.
|
|
|
|
|
*/
|
|
|
|
|
export async function POST(req: Request) {
|
|
|
|
|
let body: { sourcePath?: string; destinationDir?: string };
|
|
|
|
|
try {
|
|
|
|
|
body = await req.json();
|
|
|
|
|
} catch {
|
|
|
|
|
return Response.json({ error: "Invalid JSON body" }, { status: 400 });
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const { sourcePath, destinationDir } = body;
|
|
|
|
|
if (!sourcePath || typeof sourcePath !== "string" || !destinationDir || typeof destinationDir !== "string") {
|
|
|
|
|
return Response.json(
|
|
|
|
|
{ error: "Missing 'sourcePath' and 'destinationDir' fields" },
|
|
|
|
|
{ status: 400 },
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (isSystemFile(sourcePath)) {
|
|
|
|
|
return Response.json(
|
|
|
|
|
{ error: "Cannot move system file" },
|
|
|
|
|
{ status: 403 },
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const srcAbs = safeResolvePath(sourcePath);
|
|
|
|
|
if (!srcAbs) {
|
|
|
|
|
return Response.json(
|
|
|
|
|
{ error: "Source not found or path traversal rejected" },
|
|
|
|
|
{ status: 404 },
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const destDirAbs = safeResolvePath(destinationDir);
|
|
|
|
|
if (!destDirAbs) {
|
|
|
|
|
return Response.json(
|
|
|
|
|
{ error: "Destination not found or path traversal rejected" },
|
|
|
|
|
{ status: 404 },
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Destination must be a directory
|
|
|
|
|
if (!statSync(destDirAbs).isDirectory()) {
|
|
|
|
|
return Response.json(
|
|
|
|
|
{ error: "Destination is not a directory" },
|
|
|
|
|
{ status: 400 },
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Prevent moving a folder into itself or its children
|
|
|
|
|
const srcAbsNorm = srcAbs + "/";
|
|
|
|
|
if (destDirAbs.startsWith(srcAbsNorm) || destDirAbs === srcAbs) {
|
|
|
|
|
return Response.json(
|
|
|
|
|
{ error: "Cannot move a folder into itself" },
|
|
|
|
|
{ status: 400 },
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const itemName = basename(srcAbs);
|
|
|
|
|
const destAbs = join(destDirAbs, itemName);
|
|
|
|
|
|
|
|
|
|
if (existsSync(destAbs)) {
|
|
|
|
|
return Response.json(
|
|
|
|
|
{ error: `'${itemName}' already exists in destination` },
|
|
|
|
|
{ status: 409 },
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Build new relative path
|
|
|
|
|
const newRelPath = destinationDir === "." ? itemName : `${destinationDir}/${itemName}`;
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
renameSync(srcAbs, destAbs);
|
|
|
|
|
return Response.json({ ok: true, oldPath: sourcePath, newPath: newRelPath });
|
|
|
|
|
} catch (err) {
|
|
|
|
|
return Response.json(
|
|
|
|
|
{ error: err instanceof Error ? err.message : "Move failed" },
|
|
|
|
|
{ status: 500 },
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|