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");
|
throw new Error(err.error || "Failed to fetch projects");
|
||||||
}
|
}
|
||||||
const data = await res.json();
|
const data = await res.json();
|
||||||
setProjects(data.projects || []);
|
const loaded: ProjectWithStats[] = data.projects || [];
|
||||||
|
setProjects(loaded);
|
||||||
setError(null);
|
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) {
|
} catch (err: unknown) {
|
||||||
setError(err instanceof Error ? err.message : "Unknown error");
|
setError(err instanceof Error ? err.message : "Unknown error");
|
||||||
} finally {
|
} 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 });
|
||||||
|
}
|
||||||
@@ -85,6 +85,8 @@ export async function provisionTheiaWorkspace(
|
|||||||
{ name: 'VIBN_API_URL', value: VIBN_URL },
|
{ name: 'VIBN_API_URL', value: VIBN_URL },
|
||||||
{ name: 'GITEA_REPO', value: giteaRepo ?? '' },
|
{ name: 'GITEA_REPO', value: giteaRepo ?? '' },
|
||||||
{ name: 'GITEA_API_URL', value: process.env.GITEA_API_URL ?? 'https://git.vibnai.com' },
|
{ name: 'GITEA_API_URL', value: process.env.GITEA_API_URL ?? 'https://git.vibnai.com' },
|
||||||
|
// Token lets the startup script clone and push to the project's repo
|
||||||
|
{ name: 'GITEA_TOKEN', value: process.env.GITEA_API_TOKEN ?? '' },
|
||||||
],
|
],
|
||||||
// 5 minute startup timeout — Theia needs time to initialise
|
// 5 minute startup timeout — Theia needs time to initialise
|
||||||
startupProbe: {
|
startupProbe: {
|
||||||
|
|||||||
Reference in New Issue
Block a user