server.ts 2.8 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394
  1. import 'reflect-metadata'
  2. import 'newrelic'
  3. import * as Sentry from '@sentry/node'
  4. import '../src/Controller/HealthCheckController'
  5. import '../src/Controller/RevisionsController'
  6. import '../src/Controller/ItemsController'
  7. import * as helmet from 'helmet'
  8. import * as cors from 'cors'
  9. import { urlencoded, json, 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-SSJS-Version', container.get(TYPES.VERSION))
  23. next()
  24. })
  25. /* eslint-disable */
  26. app.use(helmet({
  27. contentSecurityPolicy: {
  28. directives: {
  29. defaultSrc: ["https: 'self'"],
  30. baseUri: ["'self'"],
  31. childSrc: ["*", "blob:"],
  32. connectSrc: ["*"],
  33. fontSrc: ["*", "'self'"],
  34. formAction: ["'self'"],
  35. frameAncestors: ["*", "*.standardnotes.org"],
  36. frameSrc: ["*", "blob:"],
  37. imgSrc: ["'self'", "*", "data:"],
  38. manifestSrc: ["'self'"],
  39. mediaSrc: ["'self'"],
  40. objectSrc: ["'self'"],
  41. scriptSrc: ["'self'"],
  42. styleSrc: ["'self'"]
  43. }
  44. }
  45. }))
  46. /* eslint-enable */
  47. app.use(json({ limit: '50mb' }))
  48. app.use(urlencoded({ extended: true, limit: '50mb', parameterLimit: 5000 }))
  49. app.use(cors())
  50. if (env.get('SENTRY_DSN', true)) {
  51. Sentry.init({
  52. dsn: env.get('SENTRY_DSN'),
  53. integrations: [new Sentry.Integrations.Http({ tracing: false, breadcrumbs: true })],
  54. tracesSampleRate: 0,
  55. })
  56. app.use(Sentry.Handlers.requestHandler() as RequestHandler)
  57. }
  58. })
  59. const logger: winston.Logger = container.get(TYPES.Logger)
  60. server.setErrorConfig((app) => {
  61. if (env.get('SENTRY_DSN', true)) {
  62. app.use(Sentry.Handlers.errorHandler() as ErrorRequestHandler)
  63. }
  64. app.use((error: Record<string, unknown>, _request: Request, response: Response, _next: NextFunction) => {
  65. logger.error(error.stack)
  66. response.status(500).send({
  67. error: {
  68. message:
  69. "Unfortunately, we couldn't handle your request. Please try again or contact our support if the error persists.",
  70. },
  71. })
  72. })
  73. })
  74. const serverInstance = server.build()
  75. serverInstance.listen(env.get('PORT'))
  76. logger.info(`Server started on port ${process.env.PORT}`)
  77. })