mirror of
https://github.com/Finsys/dockhand.git
synced 2026-06-17 19:09:33 +03:00
109 lines
3.1 KiB
TypeScript
109 lines
3.1 KiB
TypeScript
import type { RequestHandler } from '@sveltejs/kit';
|
|
import { json } from '@sveltejs/kit';
|
|
import { authorize } from '$lib/server/authorize';
|
|
import type { TemplateItem } from '../+server';
|
|
|
|
function generateContainerCompose(template: TemplateItem): string {
|
|
const name = template.title.toLowerCase().replace(/[^a-z0-9-]/g, '-').replace(/-+/g, '-');
|
|
const lines: string[] = ['services:', ` ${name}:`];
|
|
|
|
if (template.image) {
|
|
lines.push(` image: ${template.image}`);
|
|
}
|
|
|
|
if (template.restartPolicy) {
|
|
lines.push(` restart: ${template.restartPolicy}`);
|
|
}
|
|
|
|
if (template.network) {
|
|
lines.push(` network_mode: ${template.network}`);
|
|
}
|
|
|
|
if (template.ports && template.ports.length > 0 && !template.network) {
|
|
lines.push(' ports:');
|
|
for (const port of template.ports) {
|
|
lines.push(` - "${port}"`);
|
|
}
|
|
}
|
|
|
|
if (template.volumes && template.volumes.length > 0) {
|
|
lines.push(' volumes:');
|
|
for (const vol of template.volumes) {
|
|
lines.push(` - ${vol.bind}:${vol.container}`);
|
|
}
|
|
}
|
|
|
|
if (template.env && template.env.length > 0) {
|
|
lines.push(' environment:');
|
|
for (const env of template.env) {
|
|
const value = env.default || '';
|
|
lines.push(` - ${env.name}=${value}`);
|
|
}
|
|
}
|
|
|
|
return lines.join('\n') + '\n';
|
|
}
|
|
|
|
async function fetchStackCompose(repository: { url: string; stackfile: string }): Promise<string> {
|
|
// Convert GitHub URL to raw content URL
|
|
let rawUrl = '';
|
|
const githubMatch = repository.url.match(/github\.com\/([^/]+\/[^/]+)/);
|
|
|
|
if (githubMatch) {
|
|
const repo = githubMatch[1].replace(/\.git$/, '');
|
|
// Try main branch first, then master
|
|
for (const branch of ['main', 'master']) {
|
|
const url = `https://raw.githubusercontent.com/${repo}/${branch}/${repository.stackfile}`;
|
|
try {
|
|
const response = await fetch(url, { signal: AbortSignal.timeout(10000) });
|
|
if (response.ok) {
|
|
return await response.text();
|
|
}
|
|
} catch {
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Try the URL directly as fallback
|
|
if (!rawUrl) {
|
|
rawUrl = repository.url.endsWith('/')
|
|
? `${repository.url}${repository.stackfile}`
|
|
: `${repository.url}/${repository.stackfile}`;
|
|
}
|
|
|
|
const response = await fetch(rawUrl, { signal: AbortSignal.timeout(10000) });
|
|
if (!response.ok) {
|
|
throw new Error(`Failed to fetch compose file: ${response.status}`);
|
|
}
|
|
return await response.text();
|
|
}
|
|
|
|
export const POST: RequestHandler = async ({ request, cookies }) => {
|
|
const auth = await authorize(cookies);
|
|
if (auth.authEnabled && !await auth.can('templates', 'deploy')) {
|
|
return json({ error: 'Permission denied' }, { status: 403 });
|
|
}
|
|
|
|
try {
|
|
const { template } = await request.json() as { template: TemplateItem };
|
|
|
|
if (!template) {
|
|
return json({ error: 'Template is required' }, { status: 400 });
|
|
}
|
|
|
|
let compose: string;
|
|
|
|
if (template.type === 'stack' && template.repository) {
|
|
compose = await fetchStackCompose(template.repository);
|
|
} else {
|
|
compose = generateContainerCompose(template);
|
|
}
|
|
|
|
return json({ compose });
|
|
} catch (error) {
|
|
const message = error instanceof Error ? error.message : 'Failed to generate compose';
|
|
return json({ error: message }, { status: 500 });
|
|
}
|
|
};
|