File size: 3,007 Bytes
8f7dc55
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
/**
 * Aplicacion Express principal — configuracion de middlewares y montaje de rutas.
 *
 * Middlewares aplicados (en orden):
 *   1. helmet()            — headers de seguridad (X-Frame-Options, HSTS, etc.).
 *   2. cors()              — CORS con origen configurable (CORS_ORIGIN).
 *   3. rateLimit()         — 200 peticiones / 15 min por IP.
 *   4. express.json()      — parseo de JSON con limite de 1 MB.
 *
 * Rutas REST montadas bajo /api/v1:
 *   - /auth       → login, perfil (auth.routes.js)
 *   - /markets    → listado y detalle de mercados (markets.routes.js)
 *   - /markets    → senales IA por mercado (signals.routes.js, subruta)
 *   - /positions  → simulador de posiciones virtuales (positions.routes.js)
 *   - /watchlist  → lista de seguimiento (watchlist.routes.js)
 *   - /alerts     → historial de alertas (alerts.routes.js)
 *   - /health     → healthcheck basico
 *
 * Manejo de errores:
 *   - notFound      → 404 para rutas no definidas.
 *   - errorHandler  → 500 generico en produccion, detalles en desarrollo.
 */

import express from 'express';
import cors from 'cors';
import helmet from 'helmet';
import rateLimit from 'express-rate-limit';
import { config } from './config.js';
import { ok } from './utils/apiResponse.js';
import authRoutes from './auth/auth.routes.js';
import marketsRoutes from './markets/markets.routes.js';
import signalsRoutes from './signals/signals.routes.js';
import positionsRoutes from './positions/positions.routes.js';
import watchlistRoutes from './watchlist/watchlist.routes.js';
import alertsRoutes from './alerts/alerts.routes.js';
import statsRoutes from './stats/stats.routes.js';
import { notFound } from './middlewares/notFound.js';
import { errorHandler } from './middlewares/errorHandler.js';

const app = express();

app.use(helmet());
app.use(cors({ origin: config.CORS_ORIGIN, credentials: true }));

// Rate limit: muy permisivo en desarrollo, restrictivo en producción
const rateLimitMax = config.NODE_ENV === 'production' ? 200 : 5000;
app.use(
  rateLimit({
    windowMs: 15 * 60 * 1000,
    max: rateLimitMax,
    standardHeaders: true,
    legacyHeaders: false,
    message: { ok: false, error: { code: 'TOO_MANY_REQUESTS', message: 'Rate limit exceeded' } },
  }),
);
app.use(express.json({ limit: '1mb' }));

app.get('/api/v1/health', (_req, res) => ok(res, { status: 'up' }));
app.use('/api/v1/auth', authRoutes);
app.use('/api/v1/markets', marketsRoutes);
app.use('/api/v1/markets', signalsRoutes);
app.use('/api/v1/positions', positionsRoutes);
app.use('/api/v1/watchlist', watchlistRoutes);
app.use('/api/v1/alerts', alertsRoutes);
app.use('/api/v1/stats', statsRoutes);

// Servir frontend estático en producción (HuggingFace Spaces)
if (config.NODE_ENV === 'production') {
  app.use(express.static('../frontend/dist'));
  app.get('*', (_req, res) => {
    res.sendFile('index.html', { root: '../frontend/dist' });
  });
}

app.use(notFound);
app.use(errorHandler);

export default app;