server.ts 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  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. // eslint-disable-next-line @typescript-eslint/no-var-requires
  29. const robots = require('express-robots-txt')
  30. import { InversifyExpressServer } from 'inversify-express-utils'
  31. import { ContainerConfigLoader } from '../src/Bootstrap/Container'
  32. import { TYPES } from '../src/Bootstrap/Types'
  33. import { Env } from '../src/Bootstrap/Env'
  34. import { IncomingMessage, ServerResponse } from 'http'
  35. const container = new ContainerConfigLoader()
  36. void container.load().then((container) => {
  37. const env: Env = new Env()
  38. env.load()
  39. const server = new InversifyExpressServer(container)
  40. server.setConfig((app) => {
  41. app.use((_request: Request, response: Response, next: NextFunction) => {
  42. response.setHeader('X-API-Gateway-Version', container.get(TYPES.ApiGateway_VERSION))
  43. next()
  44. })
  45. app.use(
  46. helmet({
  47. contentSecurityPolicy: {
  48. directives: {
  49. defaultSrc: ["https: 'self'"],
  50. baseUri: ["'self'"],
  51. childSrc: ['*', 'blob:'],
  52. connectSrc: ['*'],
  53. fontSrc: ['*', "'self'"],
  54. formAction: ["'self'"],
  55. frameAncestors: ['*', '*.standardnotes.org', '*.standardnotes.com'],
  56. frameSrc: ['*', 'blob:'],
  57. imgSrc: ["'self'", '*', 'data:'],
  58. manifestSrc: ["'self'"],
  59. mediaSrc: ["'self'"],
  60. objectSrc: ["'self'"],
  61. scriptSrc: ["'self'"],
  62. styleSrc: ["'self'"],
  63. },
  64. },
  65. }),
  66. )
  67. app.use(
  68. json({
  69. limit: '50mb',
  70. verify: (_req: IncomingMessage, _res: ServerResponse, buf: Buffer, encoding: string): void => {
  71. try {
  72. JSON.parse(buf.toString(encoding as BufferEncoding))
  73. } catch (error) {
  74. logger.error(`Invalid JSON: ${(error as Error).message}. Request body: ${buf.toString()}`)
  75. }
  76. },
  77. }),
  78. )
  79. app.use(
  80. text({
  81. type: ['text/plain', 'application/x-www-form-urlencoded', 'application/x-www-form-urlencoded; charset=utf-8'],
  82. }),
  83. )
  84. app.use(cors())
  85. app.use(
  86. robots({
  87. UserAgent: '*',
  88. Disallow: '/',
  89. }),
  90. )
  91. })
  92. const logger: winston.Logger = container.get(TYPES.ApiGateway_Logger)
  93. server.setErrorConfig((app) => {
  94. app.use((error: Record<string, unknown>, request: Request, response: Response, _next: NextFunction) => {
  95. logger.error(
  96. `[URL: |${request.method}| ${request.url}][SNJS: ${request.headers['x-snjs-version']}][Application: ${request.headers['x-application-version']}] Error thrown: ${error.stack}`,
  97. )
  98. logger.debug(
  99. `[URL: |${request.method}| ${request.url}][SNJS: ${request.headers['x-snjs-version']}][Application: ${
  100. request.headers['x-application-version']
  101. }] Request body: ${JSON.stringify(request.body)}`,
  102. )
  103. response.status(500).send({
  104. error: {
  105. message:
  106. "Unfortunately, we couldn't handle your request. Please try again or contact our support if the error persists.",
  107. },
  108. })
  109. })
  110. })
  111. const serverInstance = server.build().listen(env.get('PORT'))
  112. const keepAliveTimeout = env.get('KEEP_ALIVE_TIMEOUT', true) ? +env.get('KEEP_ALIVE_TIMEOUT', true) : 5000
  113. serverInstance.keepAliveTimeout = keepAliveTimeout
  114. process.on('SIGTERM', () => {
  115. logger.info('SIGTERM signal received: closing HTTP server')
  116. serverInstance.close(() => {
  117. logger.info('HTTP server closed')
  118. })
  119. })
  120. logger.info(`Server started on port ${process.env.PORT}`)
  121. })