server.ts 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  1. import 'reflect-metadata'
  2. import * as busboy from 'connect-busboy'
  3. import '../src/Infra/InversifyExpress/AnnotatedFallbackController'
  4. import '../src/Infra/InversifyExpress/AnnotatedHealthCheckController'
  5. import '../src/Infra/InversifyExpress/AnnotatedFilesController'
  6. import '../src/Infra/InversifyExpress/AnnotatedSharedVaultFilesController'
  7. import helmet from 'helmet'
  8. import * as cors from 'cors'
  9. import { urlencoded, json, raw, Request, Response, NextFunction } from 'express'
  10. import * as winston from 'winston'
  11. // eslint-disable-next-line @typescript-eslint/no-var-requires
  12. const robots = require('express-robots-txt')
  13. import { InversifyExpressServer } from 'inversify-express-utils'
  14. import { ContainerConfigLoader } from '../src/Bootstrap/Container'
  15. import TYPES from '../src/Bootstrap/Types'
  16. import { Env } from '../src/Bootstrap/Env'
  17. const container = new ContainerConfigLoader('server')
  18. void container.load().then((container) => {
  19. const env: Env = new Env()
  20. env.load()
  21. const requestPayloadLimit = env.get('HTTP_REQUEST_PAYLOAD_LIMIT_MEGABYTES', true)
  22. ? `${+env.get('HTTP_REQUEST_PAYLOAD_LIMIT_MEGABYTES', true)}mb`
  23. : '50mb'
  24. const server = new InversifyExpressServer(container)
  25. server.setConfig((app) => {
  26. app.use((_request: Request, response: Response, next: NextFunction) => {
  27. response.setHeader('X-Files-Version', container.get(TYPES.Files_VERSION))
  28. next()
  29. })
  30. app.use(
  31. busboy({
  32. highWaterMark: 2 * 1024 * 1024,
  33. }),
  34. )
  35. /* eslint-disable */
  36. app.use(helmet({
  37. contentSecurityPolicy: {
  38. directives: {
  39. defaultSrc: ["https: 'self'"],
  40. baseUri: ["'self'"],
  41. childSrc: ["*", "blob:"],
  42. connectSrc: ["*"],
  43. fontSrc: ["*", "'self'"],
  44. formAction: ["'self'"],
  45. frameAncestors: ["*", "*.standardnotes.org", "*.standardnotes.com"],
  46. frameSrc: ["*", "blob:"],
  47. imgSrc: ["'self'", "*", "data:"],
  48. manifestSrc: ["'self'"],
  49. mediaSrc: ["'self'"],
  50. objectSrc: ["'self'"],
  51. scriptSrc: ["'self'"],
  52. styleSrc: ["'self'"]
  53. }
  54. }
  55. }))
  56. /* eslint-enable */
  57. app.use(json({ limit: requestPayloadLimit }))
  58. app.use(raw({ limit: requestPayloadLimit, type: 'application/octet-stream' }))
  59. app.use(urlencoded({ extended: true, limit: requestPayloadLimit }))
  60. const corsAllowedOrigins = env.get('CORS_ALLOWED_ORIGINS', true)
  61. ? env.get('CORS_ALLOWED_ORIGINS', true).split(',')
  62. : []
  63. app.use(
  64. cors({
  65. credentials: true,
  66. exposedHeaders: [
  67. 'Content-Range',
  68. 'Accept-Ranges',
  69. 'Access-Control-Allow-Credentials',
  70. 'Access-Control-Allow-Origin',
  71. ],
  72. origin: (requestOrigin: string | undefined, callback: (err: Error | null, origin?: string[]) => void) => {
  73. const originStrictModeEnabled = env.get('CORS_ORIGIN_STRICT_MODE_ENABLED', true)
  74. ? env.get('CORS_ORIGIN_STRICT_MODE_ENABLED', true) === 'true'
  75. : false
  76. if (!originStrictModeEnabled) {
  77. callback(null, [requestOrigin as string])
  78. return
  79. }
  80. const requstOriginIsNotFilled = !requestOrigin || requestOrigin === 'null'
  81. const requestOriginatesFromTheDesktopApp = requestOrigin?.startsWith('file://')
  82. const requestOriginatesFromClipperForFirefox = requestOrigin?.startsWith('moz-extension://')
  83. const requestOriginatesFromSelfHostedAppOnHttpPort = requestOrigin === 'http://localhost'
  84. const requestOriginatesFromSelfHostedAppOnCustomPort = requestOrigin?.match(/http:\/\/localhost:\d+/) !== null
  85. const requestOriginatesFromSelfHostedApp =
  86. requestOriginatesFromSelfHostedAppOnHttpPort || requestOriginatesFromSelfHostedAppOnCustomPort
  87. const requestIsWhitelisted =
  88. corsAllowedOrigins.length === 0 ||
  89. requstOriginIsNotFilled ||
  90. requestOriginatesFromTheDesktopApp ||
  91. requestOriginatesFromClipperForFirefox ||
  92. requestOriginatesFromSelfHostedApp
  93. if (requestIsWhitelisted) {
  94. callback(null, [requestOrigin as string])
  95. } else {
  96. if (corsAllowedOrigins.includes(requestOrigin)) {
  97. callback(null, [requestOrigin])
  98. } else {
  99. callback(new Error('Not allowed by CORS', { cause: 'origin not allowed' }))
  100. }
  101. }
  102. },
  103. }),
  104. )
  105. app.use(
  106. robots({
  107. UserAgent: '*',
  108. Disallow: '/',
  109. }),
  110. )
  111. })
  112. const logger: winston.Logger = container.get(TYPES.Files_Logger)
  113. server.setErrorConfig((app) => {
  114. app.use((error: Record<string, unknown>, _request: Request, response: Response, _next: NextFunction) => {
  115. logger.error(error.stack)
  116. response.status(500).send({
  117. error: {
  118. message:
  119. "Unfortunately, we couldn't handle your request. Please try again or contact our support if the error persists.",
  120. },
  121. })
  122. })
  123. })
  124. const serverInstance = server.build().listen(env.get('PORT'))
  125. const keepAliveTimeout = env.get('HTTP_KEEP_ALIVE_TIMEOUT', true) ? +env.get('HTTP_KEEP_ALIVE_TIMEOUT', true) : 5000
  126. serverInstance.keepAliveTimeout = keepAliveTimeout
  127. process.on('SIGTERM', () => {
  128. logger.info('SIGTERM signal received: closing HTTP server')
  129. serverInstance.close(() => {
  130. logger.info('HTTP server closed')
  131. })
  132. })
  133. logger.info(`Server started on port ${process.env.PORT}`)
  134. })