-
Notifications
You must be signed in to change notification settings - Fork 24
Expand file tree
/
Copy pathapp.js
More file actions
153 lines (135 loc) · 5.43 KB
/
app.js
File metadata and controls
153 lines (135 loc) · 5.43 KB
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
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
'use strict';
require('dotenv').config();
var app = require('express')();
var fs = require('fs');
var path = require('path');
var uploadDir = process.env.UPLOAD_DIRECTORY || './uploads/';
var hostname = process.env.API_HOSTNAME || 'localhost:3000';
var YAML = require('js-yaml');
var swaggerUi = require('swagger-ui-express');
var app_helper = require('./app_helper');
var createRouter = require('./api/middleware/swagger-router');
var swaggerSpec = YAML.load(fs.readFileSync('./api/swagger/swagger.yaml', 'utf8'));
var bodyParser = require('body-parser');
var api_default_port = 3000;
var express_server;
var defaultLog = app_helper.defaultLog;
// Increase postbody sizing
app.use(bodyParser.json({limit: '10mb', extended: true}));
app.use(bodyParser.urlencoded({limit: '10mb', extended: true}));
// disable powered by header
app.disable('x-powered-by');
// Enable CORS
// Reflect the requesting origin instead of '*' so that credentialed requests
// (those carrying an Authorization header) are accepted by all browsers.
app.use(function (req, res, next) {
defaultLog.info(`${req.method} ${req.url}`);
var origin = req.headers.origin || '*';
res.setHeader('Access-Control-Allow-Origin', origin);
res.setHeader('Access-Control-Allow-Methods', 'GET, POST, OPTIONS, PUT, PATCH, DELETE, HEAD');
res.setHeader('Access-Control-Allow-Headers', 'X-Requested-With,Content-Type,Authorization,responseType');
res.setHeader('Access-Control-Expose-Headers', 'x-total-count,x-pending-comment-count,x-next-comment-id');
res.setHeader('Access-Control-Allow-Credentials', true);
res.setHeader('Vary', 'Origin');
if (req.method === 'GET') {
// Authenticated requests must not be cached — admin users need fresh data
// after mutations (delete, publish, etc.). Public GETs can cache briefly.
if (req.headers.authorization) {
res.setHeader('Cache-Control', 'no-store');
} else {
res.setHeader('Cache-Control', 'max-age=60');
}
} else {
res.setHeader('Cache-Control', 'no-store');
}
// headers for zap scan issues
res.setHeader('X-XSS-Protection', '1');
res.setHeader('Strict-Transport-Security', 'max-age=63072000; includeSubDomains; preload');
if (req.method === 'OPTIONS') {
return res.sendStatus(204);
}
next();
});
// Health check — responds immediately without DB dependency.
// Used by CI smoke-test wait loop and OpenShift readiness probes.
app.get('/api/health', function (req, res) {
res.status(200).json({ status: 'ok' });
});
// Analytics proxy — forwards /analytics/* to penguin-analytics service.
// In production, nginx routes /analytics directly. This route serves local dev
// where proxy.conf.js sends /analytics to eagle-api.
var axios = require('axios');
var analyticsTarget = process.env.ANALYTICS_SERVICE_URL || 'http://localhost:3001';
app.use('/analytics', function (req, res) {
var targetUrl = analyticsTarget + '/analytics' + req.url;
axios({
method: req.method,
url: targetUrl,
data: req.body,
headers: { 'Content-Type': 'application/json' },
timeout: 5000
}).then(function (response) {
res.status(response.status).json(response.data);
}).catch(function (err) {
if (err.response) {
res.status(err.response.status).json(err.response.data);
} else {
res.status(502).json({ error: 'Analytics service unavailable' });
}
});
});
// Swagger UI — serve the API docs at /api/docs
if (hostname !== 'localhost:3000') {
swaggerSpec.schemes = ['https'];
}
swaggerSpec.host = hostname;
app.use('/api/docs', swaggerUi.serve, swaggerUi.setup(swaggerSpec));
// Auto-wired API router — reads swagger.yaml and registers one Express route per
// operation, populates req.swagger.params, enforces Bearer auth, and calls the
// matching controller function.
var controllerDirs = [
path.join(__dirname, 'api/controllers'),
path.join(__dirname, 'api/tasks')
];
app.use('/api', createRouter(swaggerSpec, controllerDirs));
// Make sure uploads directory exists
try {
if (!fs.existsSync(uploadDir)) {
fs.mkdirSync(uploadDir);
}
} catch (e) {
defaultLog.info('Couldn\'t create upload folder:', e);
}
// Skip MongoDB connection and server startup in test mode
// Tests handle their own database connection to in-memory MongoDB
if (process.env.NODE_ENV !== 'test') {
app_helper.loadMongoose().then(() => {
express_server = app.listen(api_default_port, '0.0.0.0', function() {
defaultLog.info('Started server on port ' + api_default_port);
});
}).catch(function (err) {
// loadMongoose() already retried and exited on exhaustion, but catch any
// unexpected rejection here too — crashing is better than serving 400s.
defaultLog.error('Fatal: MongoDB connection failed:', err);
process.exit(1);
});
}
// Log unhandled rejections but let the process survive.
// Mongoose auto-reconnects on transient MongoDB errors (timeouts, topology
// changes). Crashing here caused 90+ pod restarts per week in production.
process.on('unhandledRejection', function(reason) {
defaultLog.error('Unhandled Rejection:', reason);
});
function shutdown() {
if (express_server) {
console.log('Shutting down gracefully');
express_server.close(() => {
console.log('Closed out remaining connections');
process.exit(0);
});
}
}
module.exports = app;
exports.shutdown = shutdown;
exports.api_default_port = api_default_port;
exports.dbConnection = app_helper.dbConnection;