mirror of
https://github.com/standardnotes/server.git
synced 2026-01-12 06:53:23 +00:00
* chore: release latest changes * update yarn lockfile * remove stale files * fix ci env * remove mysql command overwrite * remove mysql overwrite from example * fix cookie cooldown in memory
207 lines
7.4 KiB
TypeScript
207 lines
7.4 KiB
TypeScript
import 'reflect-metadata'
|
|
|
|
import '../src/Controller/LegacyController'
|
|
import '../src/Controller/HealthCheckController'
|
|
|
|
import '../src/Controller/v1/SessionsController'
|
|
import '../src/Controller/v1/UsersController'
|
|
import '../src/Controller/v1/ActionsController'
|
|
import '../src/Controller/v1/InvoicesController'
|
|
import '../src/Controller/v1/RevisionsController'
|
|
import '../src/Controller/v1/ItemsController'
|
|
import '../src/Controller/v1/PaymentsController'
|
|
import '../src/Controller/v1/WebSocketsController'
|
|
import '../src/Controller/v1/TokensController'
|
|
import '../src/Controller/v1/OfflineController'
|
|
import '../src/Controller/v1/FilesController'
|
|
import '../src/Controller/v1/SubscriptionInvitesController'
|
|
import '../src/Controller/v1/AuthenticatorsController'
|
|
import '../src/Controller/v1/MessagesController'
|
|
import '../src/Controller/v1/SharedVaultsController'
|
|
import '../src/Controller/v1/SharedVaultInvitesController'
|
|
import '../src/Controller/v1/SharedVaultUsersController'
|
|
|
|
import '../src/Controller/v2/PaymentsControllerV2'
|
|
import '../src/Controller/v2/ActionsControllerV2'
|
|
import '../src/Controller/v2/RevisionsControllerV2'
|
|
|
|
import helmet from 'helmet'
|
|
import * as cors from 'cors'
|
|
import * as cookieParser from 'cookie-parser'
|
|
import { text, json, Request, Response, NextFunction } from 'express'
|
|
import * as winston from 'winston'
|
|
// eslint-disable-next-line @typescript-eslint/no-var-requires
|
|
const robots = require('express-robots-txt')
|
|
|
|
import { InversifyExpressServer } from 'inversify-express-utils'
|
|
import { ContainerConfigLoader } from '../src/Bootstrap/Container'
|
|
import { TYPES } from '../src/Bootstrap/Types'
|
|
import { Env } from '../src/Bootstrap/Env'
|
|
import { ResponseLocals } from '../src/Controller/ResponseLocals'
|
|
|
|
const container = new ContainerConfigLoader()
|
|
void container.load().then((container) => {
|
|
const env: Env = new Env()
|
|
env.load()
|
|
|
|
const requestPayloadLimit = env.get('HTTP_REQUEST_PAYLOAD_LIMIT_MEGABYTES', true)
|
|
? `${+env.get('HTTP_REQUEST_PAYLOAD_LIMIT_MEGABYTES', true)}mb`
|
|
: '50mb'
|
|
|
|
const logger: winston.Logger = container.get(TYPES.ApiGateway_Logger)
|
|
|
|
const server = new InversifyExpressServer(container)
|
|
|
|
server.setConfig((app) => {
|
|
app.use((request: Request, _response: Response, next: NextFunction) => {
|
|
if (request.hostname.includes('standardnotes.org')) {
|
|
logger.warn('Request is using deprecated domain', {
|
|
origin: request.headers.origin,
|
|
method: request.method,
|
|
url: request.url,
|
|
snjs: request.headers['x-snjs-version'],
|
|
application: request.headers['x-application-version'],
|
|
})
|
|
}
|
|
|
|
next()
|
|
})
|
|
app.use((_request: Request, response: Response, next: NextFunction) => {
|
|
response.setHeader('X-API-Gateway-Version', container.get(TYPES.ApiGateway_VERSION))
|
|
next()
|
|
})
|
|
app.use(
|
|
helmet({
|
|
contentSecurityPolicy: {
|
|
directives: {
|
|
defaultSrc: ["https: 'self'"],
|
|
baseUri: ["'self'"],
|
|
childSrc: ['*', 'blob:'],
|
|
connectSrc: ['*'],
|
|
fontSrc: ['*', "'self'"],
|
|
formAction: ["'self'"],
|
|
frameAncestors: ['*', '*.standardnotes.org', '*.standardnotes.com'],
|
|
frameSrc: ['*', 'blob:'],
|
|
imgSrc: ["'self'", '*', 'data:'],
|
|
manifestSrc: ["'self'"],
|
|
mediaSrc: ["'self'"],
|
|
objectSrc: ["'self'"],
|
|
scriptSrc: ["'self'"],
|
|
styleSrc: ["'self'"],
|
|
},
|
|
},
|
|
}),
|
|
)
|
|
|
|
app.use(cookieParser())
|
|
|
|
app.use(json({ limit: requestPayloadLimit }))
|
|
app.use(
|
|
text({
|
|
type: ['text/plain', 'application/x-www-form-urlencoded', 'application/x-www-form-urlencoded; charset=utf-8'],
|
|
}),
|
|
)
|
|
const corsAllowedOrigins = container.get<string[]>(TYPES.ApiGateway_CORS_ALLOWED_ORIGINS)
|
|
app.use(
|
|
cors({
|
|
credentials: true,
|
|
exposedHeaders: ['x-captcha-required'],
|
|
origin: (requestOrigin: string | undefined, callback: (err: Error | null, origin?: string[]) => void) => {
|
|
const originStrictModeEnabled = env.get('CORS_ORIGIN_STRICT_MODE_ENABLED', true)
|
|
? env.get('CORS_ORIGIN_STRICT_MODE_ENABLED', true) === 'true'
|
|
: false
|
|
|
|
if (!originStrictModeEnabled) {
|
|
callback(null, [requestOrigin as string])
|
|
|
|
return
|
|
}
|
|
|
|
const requstOriginIsNotFilled = !requestOrigin || requestOrigin === 'null'
|
|
const requestOriginatesFromTheDesktopApp = requestOrigin?.startsWith('file://')
|
|
const requestOriginatesFromClipperForFirefox = requestOrigin?.startsWith('moz-extension://')
|
|
const requestOriginatesFromSelfHostedAppOnHttpPort = requestOrigin === 'http://localhost'
|
|
const requestOriginatesFromSelfHostedAppOnCustomPort = requestOrigin?.match(/http:\/\/localhost:\d+/) !== null
|
|
const requestOriginatesFromSelfHostedApp =
|
|
requestOriginatesFromSelfHostedAppOnHttpPort || requestOriginatesFromSelfHostedAppOnCustomPort
|
|
|
|
const requestIsWhitelisted =
|
|
corsAllowedOrigins.length === 0 ||
|
|
requstOriginIsNotFilled ||
|
|
requestOriginatesFromTheDesktopApp ||
|
|
requestOriginatesFromClipperForFirefox ||
|
|
requestOriginatesFromSelfHostedApp
|
|
|
|
if (requestIsWhitelisted) {
|
|
callback(null, [requestOrigin as string])
|
|
} else {
|
|
if (corsAllowedOrigins.includes(requestOrigin)) {
|
|
callback(null, [requestOrigin])
|
|
} else {
|
|
callback(new Error('Not allowed by CORS', { cause: 'origin not allowed' }))
|
|
}
|
|
}
|
|
},
|
|
}),
|
|
)
|
|
app.use(
|
|
robots({
|
|
UserAgent: '*',
|
|
Disallow: '/',
|
|
}),
|
|
)
|
|
})
|
|
|
|
server.setErrorConfig((app) => {
|
|
app.use((error: Record<string, unknown>, request: Request, response: Response, _next: NextFunction) => {
|
|
const locals = response.locals as ResponseLocals
|
|
|
|
logger.error(`${error.stack}`, {
|
|
origin: request.headers.origin,
|
|
codeTag: 'server.ts',
|
|
method: request.method,
|
|
url: request.url,
|
|
snjs: request.headers['x-snjs-version'],
|
|
application: request.headers['x-application-version'],
|
|
userId: locals.user ? locals.user.uuid : undefined,
|
|
})
|
|
logger.debug(
|
|
`[URL: |${request.method}| ${request.url}][SNJS: ${request.headers['x-snjs-version']}][Application: ${
|
|
request.headers['x-application-version']
|
|
}] Request body: ${JSON.stringify(request.body)}`,
|
|
)
|
|
|
|
if ('type' in error && error.type === 'entity.too.large') {
|
|
response.status(413).send({
|
|
error: {
|
|
message: 'The request payload is too large.',
|
|
},
|
|
})
|
|
|
|
return
|
|
}
|
|
|
|
response.status(500).send({
|
|
error: {
|
|
message:
|
|
"Unfortunately, we couldn't handle your request. Please try again or contact our support if the error persists.",
|
|
},
|
|
})
|
|
})
|
|
})
|
|
|
|
const serverInstance = server.build().listen(env.get('PORT'))
|
|
|
|
const keepAliveTimeout = env.get('HTTP_KEEP_ALIVE_TIMEOUT', true) ? +env.get('HTTP_KEEP_ALIVE_TIMEOUT', true) : 5000
|
|
|
|
serverInstance.keepAliveTimeout = keepAliveTimeout
|
|
|
|
process.on('SIGTERM', () => {
|
|
logger.info('SIGTERM signal received: closing HTTP server')
|
|
serverInstance.close(() => {
|
|
logger.info('HTTP server closed')
|
|
})
|
|
})
|
|
|
|
logger.info(`Server started on port ${process.env.PORT}`)
|
|
})
|