platform/pods/server/src/__start.ts
Artyom Savchenko 885e20e8de
Some checks are pending
CI / test (push) Blocked by required conditions
CI / uitest (push) Waiting to run
CI / build (push) Waiting to run
CI / svelte-check (push) Blocked by required conditions
CI / formatting (push) Blocked by required conditions
CI / uitest-pg (push) Waiting to run
CI / uitest-qms (push) Waiting to run
CI / uitest-workspaces (push) Waiting to run
CI / docker-build (push) Blocked by required conditions
CI / dist-build (push) Blocked by required conditions
UBERF-9551: Add web push URL (#8133)
Signed-off-by: Artem Savchenko <armisav@gmail.com>
2025-03-04 19:43:06 +07:00

139 lines
4.6 KiB
TypeScript

//
// Copyright © 2023 Hardcore Engineering Inc
//
// Add this to the VERY top of the first file loaded in your app
import { Analytics } from '@hcengineering/analytics'
import { SplitLogger, configureAnalytics } from '@hcengineering/analytics-service'
import contactPlugin from '@hcengineering/contact'
import { MeasureMetricsContext, newMetrics, setOperationLogProfiling } from '@hcengineering/core'
import { setMetadata } from '@hcengineering/platform'
import { serverConfigFromEnv } from '@hcengineering/server'
import serverAiBot from '@hcengineering/server-ai-bot'
import serverCalendar from '@hcengineering/server-calendar'
import serverCore, {
type ConnectionSocket,
type Session,
type StorageConfiguration,
type UserStatistics,
type Workspace,
type WorkspaceStatistics,
initStatisticsContext,
loadBrandingMap
} from '@hcengineering/server-core'
import serverNotification from '@hcengineering/server-notification'
import { storageConfigFromEnv } from '@hcengineering/server-storage'
import serverTelegram from '@hcengineering/server-telegram'
import serverToken from '@hcengineering/server-token'
import { startHttpServer } from '@hcengineering/server-ws'
import { join } from 'path'
import { start } from '.'
import { profileStart, profileStop } from './inspector'
import { setDBExtraOptions } from '@hcengineering/postgres'
configureAnalytics(process.env.SENTRY_DSN, {})
Analytics.setTag('application', 'transactor')
let getUsers: () => WorkspaceStatistics[] = () => {
return []
}
// Force create server metrics context with proper logging
const metricsContext = initStatisticsContext('transactor', {
getUsers: (): WorkspaceStatistics[] => {
return getUsers()
},
factory: () =>
new MeasureMetricsContext(
'server',
{},
{},
newMetrics(),
new SplitLogger('server', {
root: join(process.cwd(), 'logs'),
enableConsole: (process.env.ENABLE_CONSOLE ?? 'true') === 'true'
})
)
})
setOperationLogProfiling(process.env.OPERATION_PROFILING === 'true')
const config = serverConfigFromEnv()
const storageConfig: StorageConfiguration = storageConfigFromEnv()
const usePrepare = (process.env.DB_PREPARE ?? 'true') === 'true'
setDBExtraOptions({
prepare: usePrepare // We override defaults
})
const lastNameFirst = process.env.LAST_NAME_FIRST === 'true'
setMetadata(contactPlugin.metadata.LastNameFirst, lastNameFirst)
setMetadata(serverCore.metadata.FrontUrl, config.frontUrl)
setMetadata(serverCore.metadata.FilesUrl, config.filesUrl)
setMetadata(serverToken.metadata.Secret, config.serverSecret)
setMetadata(serverNotification.metadata.SesUrl, config.sesUrl ?? '')
setMetadata(serverNotification.metadata.SesAuthToken, config.sesAuthToken)
setMetadata(serverNotification.metadata.WebPushUrl, config.webPushUrl ?? config.sesUrl)
setMetadata(serverTelegram.metadata.BotUrl, process.env.TELEGRAM_BOT_URL)
setMetadata(serverAiBot.metadata.EndpointURL, process.env.AI_BOT_URL)
setMetadata(serverCalendar.metadata.EndpointURL, process.env.CALENDAR_URL)
const { shutdown, sessionManager } = start(metricsContext, config.dbUrl, {
fulltextUrl: config.fulltextUrl,
storageConfig,
port: config.serverPort,
serverFactory: startHttpServer,
brandingMap: loadBrandingMap(config.brandingPath),
accountsUrl: config.accountsUrl,
enableCompression: config.enableCompression,
profiling: {
start: profileStart,
stop: profileStop
},
mongoUrl: config.mongoUrl
})
const entryToUserStats = (session: Session, socket: ConnectionSocket): UserStatistics => {
return {
current: session.current,
mins5: session.mins5,
userId: session.getUser(),
sessionId: socket.id,
total: session.total,
data: socket.data
}
}
const workspaceToWorkspaceStats = (ws: Workspace): WorkspaceStatistics => {
return {
clientsTotal: new Set(Array.from(ws.sessions.values()).map((it) => it.session.getUser())).size,
sessionsTotal: ws.sessions.size,
workspaceName: ws.workspaceName,
wsId: ws.workspaceUuid,
sessions: Array.from(ws.sessions.values()).map((it) => entryToUserStats(it.session, it.socket))
}
}
getUsers = () => {
return Array.from(sessionManager.workspaces.values()).map((it) => workspaceToWorkspaceStats(it))
}
const close = (): void => {
console.trace('Exiting from server')
console.log('Shutdown request accepted')
void shutdown().then(() => {
process.exit(0)
})
}
process.on('unhandledRejection', (reason, promise) => {
metricsContext.error('Unhandled Rejection at:', { reason, promise })
})
global.process.on('uncaughtException', (error, origin) => {
metricsContext.error('Uncaught Exception at:', { origin, error })
})
process.on('SIGINT', close)
process.on('SIGTERM', close)