server.ts 3.5 KB

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