server.js 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. const nextApp = require('next');
  2. const express = require('express');
  3. const helmet = require('helmet');
  4. const morgan = require('morgan');
  5. const cors = require('cors');
  6. const Raven = require('raven');
  7. const cookieParser = require('cookie-parser');
  8. const bodyParser = require('body-parser');
  9. const passport = require('passport');
  10. const { validateBody, validationCriterias } = require('./controllers/validateBodyController');
  11. const auth = require('./controllers/authController');
  12. const url = require('./controllers/urlController');
  13. const config = require('./config');
  14. require('./passport');
  15. if (config.RAVEN_DSN) {
  16. Raven.config(config.RAVEN_DSN).install();
  17. }
  18. const catchErrors = fn => (req, res, next) =>
  19. fn(req, res, next).catch(err => {
  20. res.status(500).json({ error: 'Sorry an error ocurred. Please try again later.' });
  21. if (config.RAVEN_DSN) {
  22. Raven.captureException(err, {
  23. user: { email: req.user && req.user.email },
  24. });
  25. } else {
  26. throw new Error(err);
  27. }
  28. });
  29. const port = Number(config.PORT) || 3000;
  30. const dev = process.env.NODE_ENV !== 'production';
  31. const app = nextApp({ dir: './client', dev });
  32. const handle = app.getRequestHandler();
  33. app.prepare().then(() => {
  34. const server = express();
  35. server.use(
  36. cors({
  37. origin: '*',
  38. methods: ['GET', 'PUT', 'POST', 'DELETE', 'OPTIONS'],
  39. allowedHeaders: [
  40. 'Origin',
  41. 'X-Requested-With',
  42. 'Content-Type',
  43. 'Accept',
  44. 'Authorization',
  45. 'Access-Control-Allow-Credentials',
  46. ],
  47. credentials: true,
  48. })
  49. );
  50. server.set('trust proxy', true);
  51. server.use(helmet());
  52. if (process.env.NODE_ENV !== 'production') {
  53. server.use(morgan('dev'));
  54. }
  55. server.use(cookieParser());
  56. server.use(bodyParser.json());
  57. server.use(bodyParser.urlencoded({ extended: true }));
  58. server.use(passport.initialize());
  59. server.use(express.static('static'));
  60. server.use((req, res, next) => {
  61. req.realIp = req.headers['x-real-ip'] || req.connection.remoteAddress || '';
  62. return next();
  63. });
  64. server.use((req, res, next) => {
  65. const { headers, path } = req;
  66. if (
  67. headers.host !== config.DEFAULT_DOMAIN &&
  68. (path === '/' || url.preservedUrls.some(item => item === path.replace('/', '')))
  69. ) {
  70. return res.redirect(`http://${config.DEFAULT_DOMAIN + path}`);
  71. }
  72. return next();
  73. });
  74. /* View routes */
  75. server.get('/', (req, res) => app.render(req, res, '/'));
  76. server.get('/login', (req, res) => app.render(req, res, '/login'));
  77. server.get('/logout', (req, res) => app.render(req, res, '/logout'));
  78. server.get('/settings', (req, res) => app.render(req, res, '/settings'));
  79. server.get('/stats', (req, res) => app.render(req, res, '/stats', req.query));
  80. server.get('/terms', (req, res) => app.render(req, res, '/terms'));
  81. server.get('/reset-password/:resetPasswordToken?', catchErrors(auth.resetPassword), (req, res) =>
  82. app.render(req, res, '/reset-password', req.user)
  83. );
  84. server.get('/verify/:verificationToken?', catchErrors(auth.verify), (req, res) =>
  85. app.render(req, res, '/verify', req.user)
  86. );
  87. /* User and authentication */
  88. server.post('/api/auth/signup', validationCriterias, validateBody, catchErrors(auth.signup));
  89. server.post('/api/auth/login', validationCriterias, validateBody, auth.authLocal, auth.login);
  90. server.post('/api/auth/renew', auth.authJwt, auth.renew);
  91. server.post('/api/auth/changepassword', auth.authJwt, catchErrors(auth.changePassword));
  92. server.post('/api/auth/generateapikey', auth.authJwt, catchErrors(auth.generateApiKey));
  93. server.post('/api/auth/resetpassword', catchErrors(auth.requestPasswordReset));
  94. server.post('/api/auth/usersettings', auth.authJwt, auth.userSettings);
  95. /* URL shortener */
  96. server.post(
  97. '/api/url/submit',
  98. auth.authApikey,
  99. auth.authJwtLoose,
  100. catchErrors(auth.recaptcha),
  101. catchErrors(url.urlShortener)
  102. );
  103. server.post('/api/url/deleteurl', auth.authApikey, auth.authJwt, catchErrors(url.deleteUrl));
  104. server.post('/api/url/geturls', auth.authApikey, auth.authJwt, catchErrors(url.getUrls));
  105. server.post('/api/url/customdomain', auth.authJwt, catchErrors(url.setCustomDomain));
  106. server.delete('/api/url/customdomain', auth.authJwt, catchErrors(url.deleteCustomDomain));
  107. server.post('/api/url/stats', auth.authApikey, auth.authJwt, catchErrors(url.getStats));
  108. server.post('/api/url/requesturl', catchErrors(url.goToUrl));
  109. server.get('/:id', catchErrors(url.goToUrl), (req, res) =>
  110. app.render(req, res, '/url-password', req.protectedUrl)
  111. );
  112. server.get('*', (req, res) => handle(req, res));
  113. server.listen(port, err => {
  114. if (err) throw err;
  115. console.log(`> Ready on http://localhost:${port}`); // eslint-disable-line no-console
  116. });
  117. });