import path from 'path'
import fs from 'fs/promises'
import dotenv from 'dotenv'
import express, { Handler, Router } from 'express'
import morgan from 'morgan'
import { createServer as createViteServer } from 'vite'
import { createApiRouter } from './server/routes'
import { RenderFunction } from './shared/ssr'
import { DatabaseConnection } from './shared/database'
// Load ".env"
dotenv.config()
const config = {
isDevelopment: process.env.MODE === 'development',
port: process.env.PORT || 3000,
}
if (config.isDevelopment) {
console.log(`[Config] Running in development mode`)
console.log(`[Config] PORT = ${config.port}`)
}
async function createDevRouter(db: DatabaseConnection) {
const r = express.Router()
const vite = await createViteServer({
server: { middlewareMode: true },
appType: 'custom',
})
r.use(vite.middlewares)
r.use('*', async (req, res, next) => {
try {
const indexHtml = await fs.readFile(path.resolve('./index.html'), 'utf-8')
const transformedTemplate = await vite.transformIndexHtml(req.originalUrl, indexHtml)
// Load (to be bundled) entry point for server side rendering
const render: RenderFunction = (await vite.ssrLoadModule('./client/entry-server.tsx')).default
const { html, metadata, asyncCallbacks } = render(req.originalUrl, db)
await Promise.all(asyncCallbacks.map(fn => fn()))
const fullUrl = `${req.protocol}://${req.get('host')}${req.originalUrl}`
const metaTagsHtml =
'' +
(metadata.title ? `\n` : '') +
(metadata.description
? `\n`
: '') +
`\n`
res.send(transformedTemplate.replace('', metaTagsHtml).replace('', html))
} catch (error: any) {
vite.ssrFixStacktrace(error)
next(error)
}
})
return r
}
async function createProductionRouter(db: DatabaseConnection) {
// Load bundled entry point for server side rendering
// @ts-ignore
const render: RenderFunction = (await import('./dist/entry-server/entry-server.js')).default
const r = express.Router()
const handleSSR = async (req, res) => {
const transformedTemplate = await fs.readFile(path.resolve('./dist/entry-client/index.html'), 'utf-8')
const { html, metadata, asyncCallbacks } = render(req.originalUrl, db)
await Promise.all(asyncCallbacks.map(fn => fn()))
const fullUrl = `${req.protocol}://${req.get('host')}${req.originalUrl}`
const metaTagsHtml =
'' +
(metadata.title
? `\n`
: `\n`) +
(metadata.description
? `\n`
: `\n`) +
`\n`
const finalHtml = transformedTemplate.replace('', metaTagsHtml).replace('', html)
console.dir(finalHtml, { depth: null })
res.send(finalHtml)
}
r.get('/', handleSSR)
r.use('/', express.static('dist/entry-client'))
r.use('*', handleSSR)
return r
}
async function main() {
const app = express()
app.use(morgan('[Request] :method :url :status :response-time ms - :res[content-length]'))
const [r, db] = await createApiRouter()
app.use(process.env.BASE_URL ?? '/', r)
if (config.isDevelopment) {
app.use(process.env.BASE_URL ?? '/', await createDevRouter(db))
} else {
app.use(process.env.BASE_URL ?? '/', await createProductionRouter(db))
}
app.listen(config.port, () => {
console.log(`[Server] Listening on port ${config.port}...`)
})
}
main()