server.ts 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104
  1. import 'reflect-metadata'
  2. import 'newrelic'
  3. import * as Sentry from '@sentry/node'
  4. import * as busboy from 'connect-busboy'
  5. import '../src/Controller/HealthCheckController'
  6. import '../src/Controller/FilesController'
  7. import * as helmet from 'helmet'
  8. import * as cors from 'cors'
  9. import { urlencoded, json, raw, Request, Response, NextFunction, RequestHandler, ErrorRequestHandler } from 'express'
  10. import * as winston from 'winston'
  11. import { InversifyExpressServer } from 'inversify-express-utils'
  12. import { ContainerConfigLoader } from '../src/Bootstrap/Container'
  13. import TYPES from '../src/Bootstrap/Types'
  14. import { Env } from '../src/Bootstrap/Env'
  15. const container = new ContainerConfigLoader()
  16. void container.load().then((container) => {
  17. const env: Env = new Env()
  18. env.load()
  19. const server = new InversifyExpressServer(container)
  20. server.setConfig((app) => {
  21. app.use((_request: Request, response: Response, next: NextFunction) => {
  22. response.setHeader('X-Files-Version', container.get(TYPES.VERSION))
  23. next()
  24. })
  25. app.use(
  26. busboy({
  27. highWaterMark: 2 * 1024 * 1024,
  28. }),
  29. )
  30. /* eslint-disable */
  31. app.use(helmet({
  32. contentSecurityPolicy: {
  33. directives: {
  34. defaultSrc: ["https: 'self'"],
  35. baseUri: ["'self'"],
  36. childSrc: ["*", "blob:"],
  37. connectSrc: ["*"],
  38. fontSrc: ["*", "'self'"],
  39. formAction: ["'self'"],
  40. frameAncestors: ["*", "*.standardnotes.org", "*.standardnotes.com"],
  41. frameSrc: ["*", "blob:"],
  42. imgSrc: ["'self'", "*", "data:"],
  43. manifestSrc: ["'self'"],
  44. mediaSrc: ["'self'"],
  45. objectSrc: ["'self'"],
  46. scriptSrc: ["'self'"],
  47. styleSrc: ["'self'"]
  48. }
  49. }
  50. }))
  51. /* eslint-enable */
  52. app.use(json({ limit: '50mb' }))
  53. app.use(raw({ limit: '50mb', type: 'application/octet-stream' }))
  54. app.use(urlencoded({ extended: true, limit: '50mb' }))
  55. app.use(
  56. cors({
  57. exposedHeaders: ['Content-Range', 'Accept-Ranges'],
  58. }),
  59. )
  60. if (env.get('SENTRY_DSN', true)) {
  61. Sentry.init({
  62. dsn: env.get('SENTRY_DSN'),
  63. integrations: [new Sentry.Integrations.Http({ tracing: false, breadcrumbs: true })],
  64. tracesSampleRate: 0,
  65. })
  66. app.use(Sentry.Handlers.requestHandler() as RequestHandler)
  67. }
  68. })
  69. const logger: winston.Logger = container.get(TYPES.Logger)
  70. server.setErrorConfig((app) => {
  71. if (env.get('SENTRY_DSN', true)) {
  72. app.use(Sentry.Handlers.errorHandler() as ErrorRequestHandler)
  73. }
  74. app.use((error: Record<string, unknown>, _request: Request, response: Response, _next: NextFunction) => {
  75. logger.error(error.stack)
  76. response.status(500).send({
  77. error: {
  78. message:
  79. "Unfortunately, we couldn't handle your request. Please try again or contact our support if the error persists.",
  80. },
  81. })
  82. })
  83. })
  84. const serverInstance = server.build()
  85. serverInstance.listen(env.get('PORT'))
  86. logger.info(`Server started on port ${process.env.PORT}`)
  87. })