server.ts 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. import 'reflect-metadata'
  2. import '../src/Controller/LegacyController'
  3. import '../src/Controller/HealthCheckController'
  4. import '../src/Controller/v1/SessionsController'
  5. import '../src/Controller/v1/UsersController'
  6. import '../src/Controller/v1/ActionsController'
  7. import '../src/Controller/v1/InvoicesController'
  8. import '../src/Controller/v1/RevisionsController'
  9. import '../src/Controller/v1/ItemsController'
  10. import '../src/Controller/v1/PaymentsController'
  11. import '../src/Controller/v1/WebSocketsController'
  12. import '../src/Controller/v1/TokensController'
  13. import '../src/Controller/v1/OfflineController'
  14. import '../src/Controller/v1/FilesController'
  15. import '../src/Controller/v1/SubscriptionInvitesController'
  16. import '../src/Controller/v1/AuthenticatorsController'
  17. import '../src/Controller/v1/MessagesController'
  18. import '../src/Controller/v1/SharedVaultsController'
  19. import '../src/Controller/v1/SharedVaultInvitesController'
  20. import '../src/Controller/v1/SharedVaultUsersController'
  21. import '../src/Controller/v2/PaymentsControllerV2'
  22. import '../src/Controller/v2/ActionsControllerV2'
  23. import '../src/Controller/v2/RevisionsControllerV2'
  24. import helmet from 'helmet'
  25. import * as cors from 'cors'
  26. import { text, json, Request, Response, NextFunction } from 'express'
  27. import * as winston from 'winston'
  28. import * as AWSXRay from 'aws-xray-sdk'
  29. // eslint-disable-next-line @typescript-eslint/no-var-requires
  30. const robots = require('express-robots-txt')
  31. import { InversifyExpressServer } from 'inversify-express-utils'
  32. import { ContainerConfigLoader } from '../src/Bootstrap/Container'
  33. import { TYPES } from '../src/Bootstrap/Types'
  34. import { Env } from '../src/Bootstrap/Env'
  35. import { ServiceIdentifier } from '@standardnotes/domain-core'
  36. const container = new ContainerConfigLoader()
  37. void container.load().then((container) => {
  38. const env: Env = new Env()
  39. env.load()
  40. const isConfiguredForAWSProduction =
  41. env.get('MODE', true) !== 'home-server' && env.get('MODE', true) !== 'self-hosted'
  42. if (isConfiguredForAWSProduction) {
  43. AWSXRay.config([AWSXRay.plugins.ECSPlugin])
  44. }
  45. const server = new InversifyExpressServer(container)
  46. server.setConfig((app) => {
  47. if (isConfiguredForAWSProduction) {
  48. app.use(AWSXRay.express.openSegment(ServiceIdentifier.NAMES.ApiGateway))
  49. }
  50. app.use((_request: Request, response: Response, next: NextFunction) => {
  51. response.setHeader('X-API-Gateway-Version', container.get(TYPES.ApiGateway_VERSION))
  52. next()
  53. })
  54. app.use(
  55. helmet({
  56. contentSecurityPolicy: {
  57. directives: {
  58. defaultSrc: ["https: 'self'"],
  59. baseUri: ["'self'"],
  60. childSrc: ['*', 'blob:'],
  61. connectSrc: ['*'],
  62. fontSrc: ['*', "'self'"],
  63. formAction: ["'self'"],
  64. frameAncestors: ['*', '*.standardnotes.org', '*.standardnotes.com'],
  65. frameSrc: ['*', 'blob:'],
  66. imgSrc: ["'self'", '*', 'data:'],
  67. manifestSrc: ["'self'"],
  68. mediaSrc: ["'self'"],
  69. objectSrc: ["'self'"],
  70. scriptSrc: ["'self'"],
  71. styleSrc: ["'self'"],
  72. },
  73. },
  74. }),
  75. )
  76. app.use(json({ limit: '50mb' }))
  77. app.use(
  78. text({
  79. type: ['text/plain', 'application/x-www-form-urlencoded', 'application/x-www-form-urlencoded; charset=utf-8'],
  80. }),
  81. )
  82. app.use(cors())
  83. app.use(
  84. robots({
  85. UserAgent: '*',
  86. Disallow: '/',
  87. }),
  88. )
  89. })
  90. const logger: winston.Logger = container.get(TYPES.ApiGateway_Logger)
  91. server.setErrorConfig((app) => {
  92. app.use((error: Record<string, unknown>, _request: Request, response: Response, _next: NextFunction) => {
  93. logger.error(error.stack)
  94. response.status(500).send({
  95. error: {
  96. message:
  97. "Unfortunately, we couldn't handle your request. Please try again or contact our support if the error persists.",
  98. },
  99. })
  100. })
  101. })
  102. const serverInstance = server.build()
  103. if (isConfiguredForAWSProduction) {
  104. serverInstance.use(AWSXRay.express.closeSegment())
  105. }
  106. serverInstance.listen(env.get('PORT'))
  107. logger.info(`Server started on port ${process.env.PORT}`)
  108. })