feat: pass GITEA_TOKEN to IDE containers + prewarm on project list load
Co-authored-by: Cursor <cursoragent@cursor.com>
This commit is contained in:
@@ -126,8 +126,22 @@ export default function ProjectsPage() {
|
||||
throw new Error(err.error || "Failed to fetch projects");
|
||||
}
|
||||
const data = await res.json();
|
||||
setProjects(data.projects || []);
|
||||
const loaded: ProjectWithStats[] = data.projects || [];
|
||||
setProjects(loaded);
|
||||
setError(null);
|
||||
|
||||
// Fire-and-forget: prewarm all provisioned IDE workspaces so containers
|
||||
// are already running by the time the user clicks "Open IDE"
|
||||
const warmUrls = loaded
|
||||
.map((p) => p.theiaWorkspaceUrl)
|
||||
.filter((u): u is string => Boolean(u));
|
||||
if (warmUrls.length > 0) {
|
||||
fetch("/api/projects/prewarm", {
|
||||
method: "POST",
|
||||
headers: { "Content-Type": "application/json" },
|
||||
body: JSON.stringify({ urls: warmUrls }),
|
||||
}).catch(() => {}); // ignore errors — this is best-effort
|
||||
}
|
||||
} catch (err: unknown) {
|
||||
setError(err instanceof Error ? err.message : "Unknown error");
|
||||
} finally {
|
||||
|
||||
29
app/api/projects/prewarm/route.ts
Normal file
29
app/api/projects/prewarm/route.ts
Normal file
@@ -0,0 +1,29 @@
|
||||
import { NextRequest, NextResponse } from 'next/server';
|
||||
import { getServerSession } from 'next-auth';
|
||||
import { authOptions } from '@/lib/auth';
|
||||
import { prewarmWorkspace } from '@/lib/cloud-run-workspace';
|
||||
|
||||
/**
|
||||
* POST /api/projects/prewarm
|
||||
* Body: { urls: string[] }
|
||||
*
|
||||
* Fires warm-up requests to Cloud Run workspace URLs so containers
|
||||
* are running by the time the user clicks "Open IDE". Server-side
|
||||
* to avoid CORS issues with run.app domains.
|
||||
*/
|
||||
export async function POST(req: NextRequest) {
|
||||
const session = await getServerSession(authOptions);
|
||||
if (!session?.user) {
|
||||
return NextResponse.json({ error: 'Unauthorized' }, { status: 401 });
|
||||
}
|
||||
|
||||
const { urls } = await req.json() as { urls: string[] };
|
||||
if (!Array.isArray(urls) || urls.length === 0) {
|
||||
return NextResponse.json({ warmed: 0 });
|
||||
}
|
||||
|
||||
// Fire all prewarm pings in parallel — intentionally not awaited
|
||||
Promise.allSettled(urls.map(url => prewarmWorkspace(url))).catch(() => {});
|
||||
|
||||
return NextResponse.json({ warmed: urls.length });
|
||||
}
|
||||
Reference in New Issue
Block a user