server.ts 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. import 'reflect-metadata'
  2. import '../src/Infra/InversifyExpressUtils/AnnotatedFallbackController'
  3. import '../src/Infra/InversifyExpressUtils/AnnotatedHealthCheckController'
  4. import '../src/Infra/InversifyExpressUtils/AnnotatedItemsController'
  5. import '../src/Infra/InversifyExpressUtils/AnnotatedMessagesController'
  6. import '../src/Infra/InversifyExpressUtils/AnnotatedSharedVaultInvitesController'
  7. import '../src/Infra/InversifyExpressUtils/AnnotatedSharedVaultUsersController'
  8. import '../src/Infra/InversifyExpressUtils/AnnotatedSharedVaultsController'
  9. import helmet from 'helmet'
  10. import * as cors from 'cors'
  11. import * as grpc from '@grpc/grpc-js'
  12. import { urlencoded, json, Request, Response, NextFunction } from 'express'
  13. import * as winston from 'winston'
  14. import { InversifyExpressServer } from 'inversify-express-utils'
  15. import { MapperInterface } from '@standardnotes/domain-core'
  16. import { SyncResponse, SyncingService } from '@standardnotes/grpc'
  17. import TYPES from '../src/Bootstrap/Types'
  18. import { Env } from '../src/Bootstrap/Env'
  19. import { ContainerConfigLoader } from '../src/Bootstrap/Container'
  20. import { SyncingServer } from '../src/Infra/gRPC/SyncingServer'
  21. import { SyncItems } from '../src/Domain/UseCase/Syncing/SyncItems/SyncItems'
  22. import { SyncResponseFactoryResolverInterface } from '../src/Domain/Item/SyncResponse/SyncResponseFactoryResolverInterface'
  23. import { SyncResponse20200115 } from '../src/Domain/Item/SyncResponse/SyncResponse20200115'
  24. const container = new ContainerConfigLoader()
  25. void container.load().then((container) => {
  26. const env: Env = new Env()
  27. env.load()
  28. const server = new InversifyExpressServer(container)
  29. server.setConfig((app) => {
  30. app.use((_request: Request, response: Response, next: NextFunction) => {
  31. response.setHeader('X-SSJS-Version', container.get(TYPES.Sync_VERSION))
  32. next()
  33. })
  34. /* eslint-disable */
  35. app.use(helmet({
  36. contentSecurityPolicy: {
  37. directives: {
  38. defaultSrc: ["https: 'self'"],
  39. baseUri: ["'self'"],
  40. childSrc: ["*", "blob:"],
  41. connectSrc: ["*"],
  42. fontSrc: ["*", "'self'"],
  43. formAction: ["'self'"],
  44. frameAncestors: ["*", "*.standardnotes.org"],
  45. frameSrc: ["*", "blob:"],
  46. imgSrc: ["'self'", "*", "data:"],
  47. manifestSrc: ["'self'"],
  48. mediaSrc: ["'self'"],
  49. objectSrc: ["'self'"],
  50. scriptSrc: ["'self'"],
  51. styleSrc: ["'self'"]
  52. }
  53. }
  54. }))
  55. /* eslint-enable */
  56. app.use(json({ limit: '50mb' }))
  57. app.use(urlencoded({ extended: true, limit: '50mb', parameterLimit: 5000 }))
  58. app.use(cors())
  59. })
  60. const logger: winston.Logger = container.get(TYPES.Sync_Logger)
  61. server.setErrorConfig((app) => {
  62. app.use((error: Record<string, unknown>, _request: Request, response: Response, _next: NextFunction) => {
  63. logger.error(error.stack)
  64. response.status(500).send({
  65. error: {
  66. message:
  67. "Unfortunately, we couldn't handle your request. Please try again or contact our support if the error persists.",
  68. },
  69. })
  70. })
  71. })
  72. const serverInstance = server.build().listen(env.get('PORT'))
  73. const keepAliveTimeout = env.get('HTTP_KEEP_ALIVE_TIMEOUT', true) ? +env.get('HTTP_KEEP_ALIVE_TIMEOUT', true) : 5000
  74. serverInstance.keepAliveTimeout = keepAliveTimeout
  75. const grpcKeepAliveTime = env.get('GRPC_KEEP_ALIVE_TIME', true) ? +env.get('GRPC_KEEP_ALIVE_TIME', true) : 7_200_000
  76. const grpcKeepAliveTimeout = env.get('GRPC_KEEP_ALIVE_TIMEOUT', true)
  77. ? +env.get('GRPC_KEEP_ALIVE_TIMEOUT', true)
  78. : 20_000
  79. const grpcMaxMessageSize = env.get('GRPC_MAX_MESSAGE_SIZE', true)
  80. ? +env.get('GRPC_MAX_MESSAGE_SIZE', true)
  81. : 1024 * 1024 * 50
  82. const grpcServer = new grpc.Server({
  83. 'grpc.keepalive_time_ms': grpcKeepAliveTime,
  84. 'grpc.keepalive_timeout_ms': grpcKeepAliveTimeout,
  85. 'grpc.default_compression_algorithm': grpc.compressionAlgorithms.gzip,
  86. 'grpc.max_receive_message_length': grpcMaxMessageSize,
  87. 'grpc.max_send_message_length': grpcMaxMessageSize,
  88. })
  89. const gRPCPort = env.get('GRPC_PORT', true) ? +env.get('GRPC_PORT', true) : 50051
  90. const syncingServer = new SyncingServer(
  91. container.get<SyncItems>(TYPES.Sync_SyncItems),
  92. container.get<SyncResponseFactoryResolverInterface>(TYPES.Sync_SyncResponseFactoryResolver),
  93. container.get<MapperInterface<SyncResponse20200115, SyncResponse>>(TYPES.Sync_SyncResponseGRPCMapper),
  94. container.get<winston.Logger>(TYPES.Sync_Logger),
  95. )
  96. grpcServer.addService(SyncingService, {
  97. syncItems: syncingServer.syncItems.bind(syncingServer),
  98. })
  99. grpcServer.bindAsync(`0.0.0.0:${gRPCPort}`, grpc.ServerCredentials.createInsecure(), (error, port) => {
  100. if (error) {
  101. logger.error(`Failed to bind gRPC server: ${error.message}`)
  102. return
  103. }
  104. logger.info(`gRPC server bound on port ${port}`)
  105. grpcServer.start()
  106. logger.info('gRPC server started')
  107. })
  108. process.on('SIGTERM', () => {
  109. logger.info('SIGTERM signal received: closing HTTP server')
  110. serverInstance.close(() => {
  111. logger.info('HTTP server closed')
  112. })
  113. grpcServer.tryShutdown((error?: Error) => {
  114. if (error) {
  115. logger.error(`Failed to shutdown gRPC server: ${error.message}`)
  116. } else {
  117. logger.info('gRPC server closed')
  118. }
  119. })
  120. })
  121. logger.info(`Server started on port ${process.env.PORT}`)
  122. })