Configurar despliegue con Docker, Traefik y Authentik
All checks were successful
build-and-deploy / build-and-deploy (push) Successful in 27s

- Agregar Dockerfile para build multi-stage con Node 20
- Configurar docker-compose.yml con Traefik y Authentik exteriorlvl2
- Crear workflow de Gitea Actions para CI/CD automático
- Configurar routers público (assets) y protegido (app + APIs)
- Documentar arquitectura y proceso de despliegue
This commit is contained in:
2025-10-27 12:00:05 -06:00
commit 4c729866aa
13 changed files with 3091 additions and 0 deletions

98
server.js Normal file
View File

@@ -0,0 +1,98 @@
import express from 'express';
import cors from 'cors';
import { fileURLToPath } from 'url';
import { dirname, join } from 'path';
import { readdir } from 'fs/promises';
import archiver from 'archiver';
const __filename = fileURLToPath(import.meta.url);
const __dirname = dirname(__filename);
const app = express();
const PORT = process.env.PORT || 3001;
// Middleware
app.use(cors());
app.use(express.json());
// Servir archivos estáticos (HTML, CSS, JS)
app.use(express.static(join(__dirname, 'public')));
// Servir fotos
app.use('/photos', express.static(join(__dirname, 'photos')));
// API para listar fotos
app.get('/api/photos', async (req, res) => {
try {
const photosDir = join(__dirname, 'photos');
const files = await readdir(photosDir, { withFileTypes: true });
const photos = files
.filter(file => {
if (!file.isFile()) return false;
const ext = file.name.toLowerCase().split('.').pop();
return ['jpg', 'jpeg', 'png', 'gif', 'webp'].includes(ext);
})
.map(file => ({
name: file.name,
url: `/photos/${file.name}`
}));
res.json(photos);
} catch (error) {
console.error('Error al listar fotos:', error);
res.status(500).json({ error: 'Error al listar fotos' });
}
});
// API para descargar todas las fotos como ZIP
app.get('/api/photos/zip', async (req, res) => {
try {
const photosDir = join(__dirname, 'photos');
const files = await readdir(photosDir, { withFileTypes: true });
const photoFiles = files.filter(file => {
if (!file.isFile()) return false;
const ext = file.name.toLowerCase().split('.').pop();
return ['jpg', 'jpeg', 'png', 'gif', 'webp'].includes(ext);
});
if (photoFiles.length === 0) {
return res.status(404).json({ error: 'No hay fotos para descargar' });
}
// Configurar headers para la descarga
res.setHeader('Content-Type', 'application/zip');
res.setHeader('Content-Disposition', 'attachment; filename=fotos.zip');
// Crear archivo ZIP
const archive = archiver('zip', {
zlib: { level: 9 } // Nivel de compresión
});
// Manejar errores del archiver
archive.on('error', (err) => {
console.error('Error al crear ZIP:', err);
res.status(500).json({ error: 'Error al crear ZIP' });
});
// Pipe del archivo al response
archive.pipe(res);
// Agregar archivos al ZIP
for (const file of photoFiles) {
archive.file(join(photosDir, file.name), { name: file.name });
}
// Finalizar el archivo
await archive.finalize();
} catch (error) {
console.error('Error al generar ZIP:', error);
res.status(500).json({ error: 'Error al generar ZIP' });
}
});
app.listen(PORT, () => {
console.log(`🚀 Servidor de fotos corriendo en http://localhost:${PORT}`);
console.log(`📸 API de fotos disponible en http://localhost:${PORT}/api/photos`);
});