Add project soft-delete with workspace archival

- Add delete button (×) to sidebar project list, shown on hover
- Soft-delete: mark projects as deleted in DB instead of hard delete
- Move workspace files to /app/data/deleted/ folder on deletion
- Filter deleted projects from list query
- Auto-select next project after deleting current one
- Also includes agent prompt improvements for reverse proxy paths

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
Fam Zheng 2026-03-01 07:56:37 +00:00
parent 2df4e12d30
commit 1aa81896b5
6 changed files with 112 additions and 7 deletions

View File

@ -2,8 +2,12 @@ context compaction
rag / kb
内置一个向量数据库和kb管理能力吧kb_search管理界面简单点cpu做embedding就够
template
---
时间观察app
---
## 代码啰嗦/可精简

View File

@ -474,8 +474,13 @@ fn build_planning_prompt(project_id: &str) -> String {
- Python venv .venv/\n\
- bashgitcurluv\n\
- 访/api/projects/{0}/files/{{filename}}\n\
- 访/api/projects/{0}/app/\n\
- Web FastAPI + HTMLAPI /api/projects/{0}/app/...\n\
- 访/api/projects/{0}/app/ /\n\
\n\
\n\
- /api/projects/{0}/app/ 访 / \n\
- HTML API 使 / \n\
- fetch('todos') fetch('./todos') fetch('/todos') fetch('/api/todos')\n\
- HTML <base> \n\
\n\
使",
project_id,
@ -505,6 +510,7 @@ fn build_execution_prompt(project_id: &str) -> String {
- 使 `uv add <>` `pip install <>` \n\
- 访/api/projects/{0}/files/{{filename}}\n\
- 访/api/projects/{0}/app/ 0.0.0.0:$PORT\n\
- 访 HTML/JS fetch/XHR 使 fetch('todos') / fetch('/todos') 404\n\
\n\
使",
project_id,

View File

@ -40,7 +40,7 @@ pub fn router(state: Arc<AppState>) -> Router {
async fn list_projects(
State(state): State<Arc<AppState>>,
) -> ApiResult<Vec<Project>> {
sqlx::query_as::<_, Project>("SELECT * FROM projects ORDER BY updated_at DESC")
sqlx::query_as::<_, Project>("SELECT * FROM projects WHERE deleted = 0 ORDER BY updated_at DESC")
.fetch_all(&state.db.pool)
.await
.map(Json)
@ -109,10 +109,30 @@ async fn delete_project(
State(state): State<Arc<AppState>>,
Path(id): Path<String>,
) -> ApiResult<bool> {
sqlx::query("DELETE FROM projects WHERE id = ?")
// Soft delete: mark as deleted in DB
let result = sqlx::query("UPDATE projects SET deleted = 1, updated_at = datetime('now') WHERE id = ? AND deleted = 0")
.bind(&id)
.execute(&state.db.pool)
.await
.map(|r| Json(r.rows_affected() > 0))
.map_err(db_err)
.map_err(db_err)?;
if result.rows_affected() == 0 {
return Ok(Json(false));
}
// Move workspace to deleted folder
let src = std::path::PathBuf::from("/app/data/workspaces").join(&id);
let dst_dir = std::path::PathBuf::from("/app/data/deleted");
if src.exists() {
if let Err(e) = tokio::fs::create_dir_all(&dst_dir).await {
tracing::error!("Failed to create deleted dir: {}", e);
} else {
let dst = dst_dir.join(&id);
if let Err(e) = tokio::fs::rename(&src, &dst).await {
tracing::error!("Failed to move workspace to deleted: {}", e);
}
}
}
Ok(Json(true))
}

View File

@ -94,6 +94,13 @@ impl Database {
.execute(&self.pool)
.await;
// Migration: add deleted column to projects
let _ = sqlx::query(
"ALTER TABLE projects ADD COLUMN deleted INTEGER NOT NULL DEFAULT 0"
)
.execute(&self.pool)
.await;
sqlx::query(
"CREATE TABLE IF NOT EXISTS timers (
id TEXT PRIMARY KEY,
@ -120,6 +127,8 @@ pub struct Project {
pub description: String,
pub created_at: String,
pub updated_at: String,
#[serde(default)]
pub deleted: bool,
}
#[derive(Debug, Clone, Serialize, Deserialize, sqlx::FromRow)]

View File

@ -80,6 +80,23 @@ function onProjectUpdate(projectId: string, name: string) {
const p = projects.value.find(p => p.id === projectId)
if (p) p.name = name
}
async function onDeleteProject(id: string) {
try {
await api.deleteProject(id)
projects.value = projects.value.filter(p => p.id !== id)
if (selectedProjectId.value === id) {
selectedProjectId.value = projects.value[0]?.id ?? ''
if (selectedProjectId.value) {
history.replaceState(null, '', `/projects/${selectedProjectId.value}`)
} else {
history.replaceState(null, '', '/')
}
}
} catch (e: any) {
error.value = e.message
}
}
</script>
<template>
@ -92,6 +109,7 @@ function onProjectUpdate(projectId: string, name: string) {
:selectedId="selectedProjectId"
@select="onSelectProject"
@create="onStartCreate"
@delete="onDeleteProject"
/>
<main class="main-content">
<div v-if="error" class="error-banner" @click="error = ''">{{ error }}</div>

View File

@ -9,7 +9,15 @@ defineProps<{
const emit = defineEmits<{
select: [id: string]
create: []
delete: [id: string]
}>()
function onDelete(e: Event, id: string) {
e.stopPropagation()
if (confirm('确定删除这个项目?')) {
emit('delete', id)
}
}
</script>
<template>
@ -26,7 +34,10 @@ const emit = defineEmits<{
:class="{ active: project.id === selectedId }"
@click="emit('select', project.id)"
>
<div class="project-row">
<span class="project-name">{{ project.name }}</span>
<button class="btn-delete" @click="onDelete($event, project.id)" title="删除项目">×</button>
</div>
<span class="project-time">{{ new Date(project.updated_at).toLocaleDateString() }}</span>
</div>
</nav>
@ -96,9 +107,46 @@ const emit = defineEmits<{
border-left: 3px solid var(--accent);
}
.project-row {
display: flex;
align-items: center;
justify-content: space-between;
}
.project-name {
font-size: 14px;
font-weight: 500;
flex: 1;
min-width: 0;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.btn-delete {
display: none;
width: 20px;
height: 20px;
padding: 0;
border: none;
background: transparent;
color: var(--text-secondary);
font-size: 16px;
line-height: 1;
cursor: pointer;
border-radius: 4px;
flex-shrink: 0;
}
.btn-delete:hover {
background: var(--error, #e74c3c);
color: #fff;
}
.project-item:hover .btn-delete {
display: flex;
align-items: center;
justify-content: center;
}
.project-time {