2020-08-08 13:51:21 -06:00
|
|
|
/*
|
2021-01-11 06:58:47 -07:00
|
|
|
Copyright (C) 2021 Yarmo Mackenbach
|
2020-08-08 13:51:21 -06:00
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify it under
|
|
|
|
the terms of the GNU Affero General Public License as published by the Free
|
|
|
|
Software Foundation, either version 3 of the License, or (at your option)
|
|
|
|
any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
|
|
|
FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
|
|
|
|
details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Affero General Public License along
|
|
|
|
with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
Also add information on how to contact you by electronic and paper mail.
|
|
|
|
|
|
|
|
If your software can interact with users remotely through a computer network,
|
|
|
|
you should also make sure that it provides a way for users to get its source.
|
|
|
|
For example, if your program is a web application, its interface could display
|
|
|
|
a "Source" link that leads users to an archive of the code. There are many
|
|
|
|
ways you could offer source, and different solutions will be better for different
|
|
|
|
programs; see section 13 for the specific requirements.
|
|
|
|
|
|
|
|
You should also get your employer (if you work as a programmer) or school,
|
|
|
|
if any, to sign a "copyright disclaimer" for the program, if necessary. For
|
|
|
|
more information on this, and how to apply and follow the GNU AGPL, see <https://www.gnu.org/licenses/>.
|
|
|
|
*/
|
2022-02-27 14:59:17 -07:00
|
|
|
import express from 'express'
|
2023-09-21 14:54:37 -06:00
|
|
|
import * as httpContext from 'express-http-context2'
|
|
|
|
import { nanoid } from 'nanoid'
|
2022-02-27 14:59:17 -07:00
|
|
|
import { readFileSync } from 'fs'
|
|
|
|
import { stringReplace } from 'string-replace-middleware'
|
2022-03-15 09:23:20 -06:00
|
|
|
import * as pug from 'pug'
|
2022-11-17 14:54:13 -07:00
|
|
|
import * as dotenv from 'dotenv'
|
2022-02-27 14:59:17 -07:00
|
|
|
|
2023-03-24 05:53:33 -06:00
|
|
|
import logger from './log.js'
|
2022-11-14 13:39:26 -07:00
|
|
|
import apiRoute from './routes/api.js'
|
|
|
|
import mainRoute from './routes/main.js'
|
|
|
|
import profileRoute from './routes/profile.js'
|
|
|
|
import staticRoute from './routes/static.js'
|
|
|
|
import utilRoute from './routes/util.js'
|
2023-03-01 07:53:18 -07:00
|
|
|
dotenv.config()
|
2020-08-07 17:06:58 -06:00
|
|
|
|
2022-02-27 14:59:17 -07:00
|
|
|
const app = express()
|
2022-03-15 13:54:02 -06:00
|
|
|
const packageData = JSON.parse(readFileSync('./package.json'))
|
2020-08-14 17:25:00 -06:00
|
|
|
|
2022-03-15 15:26:10 -06:00
|
|
|
app.set('env', process.env.NODE_ENV || 'production')
|
2022-03-15 09:23:20 -06:00
|
|
|
app.engine('pug', pug.__express).set('view engine', 'pug')
|
2021-05-02 04:49:52 -06:00
|
|
|
app.set('port', process.env.PORT || 3000)
|
2022-07-13 05:38:50 -06:00
|
|
|
app.set('domain', process.env.DOMAIN)
|
2023-06-11 22:41:33 -06:00
|
|
|
app.set('scheme', process.env.SCHEME || 'https')
|
2021-05-02 04:49:52 -06:00
|
|
|
app.set('keyoxide_version', packageData.version)
|
|
|
|
app.set('onion_url', process.env.ONION_URL)
|
2020-08-14 12:16:14 -06:00
|
|
|
|
2021-06-21 03:35:59 -06:00
|
|
|
// Middlewares
|
2023-09-21 14:54:37 -06:00
|
|
|
app.use(httpContext.middleware)
|
2021-05-04 04:54:03 -06:00
|
|
|
app.use((req, res, next) => {
|
2022-11-14 12:42:50 -07:00
|
|
|
res.setHeader('Permissions-Policy', 'interest-cohort=()')
|
2023-09-21 14:54:37 -06:00
|
|
|
httpContext.set('requestId', nanoid())
|
|
|
|
httpContext.set('requestPath', req.path)
|
|
|
|
httpContext.set('requestIp', req.ip)
|
|
|
|
|
2023-09-22 01:48:55 -06:00
|
|
|
logger.info('Handle a request',
|
2023-09-21 14:54:37 -06:00
|
|
|
{ component: 'http_server', action: 'request' })
|
|
|
|
|
2022-11-14 12:42:50 -07:00
|
|
|
next()
|
2021-05-04 04:54:03 -06:00
|
|
|
})
|
|
|
|
|
2021-06-21 03:35:59 -06:00
|
|
|
if (app.get('onion_url')) {
|
2022-11-14 12:42:50 -07:00
|
|
|
app.get('/*', (req, res, next) => {
|
|
|
|
res.header('Onion-Location', app.get('onion_url'))
|
|
|
|
next()
|
|
|
|
})
|
2021-06-21 03:35:59 -06:00
|
|
|
}
|
2020-08-07 17:06:58 -06:00
|
|
|
|
2020-08-14 08:23:32 -06:00
|
|
|
app.use(stringReplace({
|
2023-06-11 22:41:33 -06:00
|
|
|
PLACEHOLDER__PROXY_HOSTNAME: process.env.PROXY_HOSTNAME || process.env.DOMAIN || 'null',
|
|
|
|
PLACEHOLDER__PROXY_SCHEME: process.env.PROXY_SCHEME || process.env.SCHEME || 'https'
|
2020-08-14 08:23:32 -06:00
|
|
|
}, {
|
2022-11-14 12:42:50 -07:00
|
|
|
contentTypeFilterRegexp: /application\/javascript/
|
2021-05-02 04:49:52 -06:00
|
|
|
}))
|
2020-08-14 08:23:32 -06:00
|
|
|
|
2021-06-21 03:35:59 -06:00
|
|
|
// Routes
|
2022-11-14 13:58:56 -07:00
|
|
|
app.use('/api', apiRoute)
|
2022-11-14 14:07:36 -07:00
|
|
|
if ((process.env.ENABLE_MAIN_MODULE ?? 'true') === 'true') {
|
2022-11-14 13:39:26 -07:00
|
|
|
app.use('/favicon.svg', express.static('./static/favicon.svg'))
|
|
|
|
app.use('/robots.txt', express.static('./static/robots.txt'))
|
|
|
|
|
|
|
|
app.use('/', mainRoute)
|
|
|
|
app.use('/static', staticRoute)
|
|
|
|
app.use('/util', utilRoute)
|
|
|
|
app.use('/', profileRoute)
|
|
|
|
}
|
2020-08-07 17:06:58 -06:00
|
|
|
|
2020-08-14 11:54:28 -06:00
|
|
|
app.listen(app.get('port'), () => {
|
2023-03-24 05:53:33 -06:00
|
|
|
logger.info(`Server listening at http://localhost:${app.get('port')}`,
|
|
|
|
{ component: 'http_server', action: 'start' })
|
2021-05-02 04:49:52 -06:00
|
|
|
})
|
2022-02-27 14:59:17 -07:00
|
|
|
|
|
|
|
export default app
|