2024-05-23 11:42:14 +02:00
|
|
|
// SPDX-FileCopyrightText: 2024 John Livingston <https://www.john-livingston.fr/>
|
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2024-04-16 18:49:23 +02:00
|
|
|
import type { RegisterServerOptions } from '@peertube/peertube-types'
|
2024-04-17 12:09:25 +02:00
|
|
|
import type { Router, Request, Response, NextFunction } from 'express'
|
2024-04-19 09:53:23 +02:00
|
|
|
import type { ExternalAuthResult } from '../../../shared/lib/types'
|
2024-04-16 18:49:23 +02:00
|
|
|
import { asyncMiddleware } from '../middlewares/async'
|
|
|
|
import { ExternalAuthOIDC } from '../external-auth/oidc'
|
2024-04-17 15:12:37 +02:00
|
|
|
import { ExternalAuthenticationError } from '../external-auth/error'
|
2024-04-17 16:35:26 +02:00
|
|
|
import { ensureUser } from '../prosody/api/manage-users'
|
2024-04-16 18:49:23 +02:00
|
|
|
|
2024-04-17 12:09:25 +02:00
|
|
|
/**
|
|
|
|
* When using a popup for OIDC, writes the HTML/Javascript to close the popup
|
|
|
|
* and send the result to the parent window.
|
|
|
|
* @param result the result to send to the parent window
|
|
|
|
*/
|
2024-04-19 09:53:23 +02:00
|
|
|
function popupResultHTML (result: ExternalAuthResult): string {
|
2024-04-17 12:09:25 +02:00
|
|
|
return `<!DOCTYPE html><html>
|
|
|
|
<body>
|
|
|
|
<noscript>Your browser must enable javascript for this page to work.</noscript>
|
|
|
|
<script>
|
|
|
|
try {
|
|
|
|
const data = ${JSON.stringify(result)};
|
2024-04-19 09:53:23 +02:00
|
|
|
if (!window.opener || !window.opener.externalAuthGetResult) {
|
2024-04-17 12:09:25 +02:00
|
|
|
throw new Error("Can't find parent window callback handler.")
|
|
|
|
}
|
2024-04-19 09:53:23 +02:00
|
|
|
window.opener.externalAuthGetResult(data);
|
2024-04-17 12:09:25 +02:00
|
|
|
window.close();
|
|
|
|
} catch (err) {
|
|
|
|
document.body.innerText = 'Error: ' + err;
|
|
|
|
}
|
|
|
|
</script>
|
|
|
|
</body>
|
|
|
|
</html> `
|
2024-04-16 18:49:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async function initOIDCRouter (options: RegisterServerOptions): Promise<Router> {
|
|
|
|
const { peertubeHelpers, getRouter } = options
|
|
|
|
const router = getRouter()
|
|
|
|
const logger = peertubeHelpers.logger
|
|
|
|
|
2024-04-22 13:03:31 +02:00
|
|
|
router.get('/:type?/connect', asyncMiddleware(
|
2024-04-16 18:49:23 +02:00
|
|
|
async (req: Request, res: Response, next: NextFunction) => {
|
2024-04-22 13:03:31 +02:00
|
|
|
const singletonType = req.params.type ?? 'custom'
|
|
|
|
logger.info('[oidc router] OIDC connect call (' + singletonType + ')')
|
2024-04-16 18:49:23 +02:00
|
|
|
try {
|
2024-04-22 13:03:31 +02:00
|
|
|
const oidc = ExternalAuthOIDC.singleton(singletonType)
|
2024-04-16 18:49:23 +02:00
|
|
|
const oidcClient = await oidc.load()
|
|
|
|
if (!oidcClient) {
|
|
|
|
throw new Error('[oidc router] External Auth OIDC not loaded yet')
|
|
|
|
}
|
|
|
|
|
2024-04-17 12:09:25 +02:00
|
|
|
const redirectUrl = await oidc.initAuthenticationProcess(req, res)
|
|
|
|
res.redirect(redirectUrl)
|
2024-04-16 18:49:23 +02:00
|
|
|
} catch (err) {
|
2024-09-09 21:21:44 +02:00
|
|
|
logger.error('[oidc router] Failed to process the OIDC connect call: ' + (err as string))
|
2024-04-16 18:49:23 +02:00
|
|
|
next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
))
|
|
|
|
|
2024-04-18 09:52:27 +02:00
|
|
|
const cbHandler = asyncMiddleware(
|
2024-04-17 12:09:25 +02:00
|
|
|
async (req: Request, res: Response, _next: NextFunction) => {
|
2024-04-22 13:03:31 +02:00
|
|
|
const singletonType = req.params.type ?? 'custom'
|
|
|
|
logger.info('[oidc router] OIDC callback call (' + singletonType + ')')
|
2024-04-16 18:49:23 +02:00
|
|
|
try {
|
2024-04-22 13:03:31 +02:00
|
|
|
const oidc = ExternalAuthOIDC.singleton(singletonType)
|
2024-04-16 18:49:23 +02:00
|
|
|
const oidcClient = await oidc.load()
|
|
|
|
if (!oidcClient) {
|
|
|
|
throw new Error('[oidc router] External Auth OIDC not loaded yet')
|
|
|
|
}
|
|
|
|
|
2024-04-17 15:12:37 +02:00
|
|
|
const externalAccountInfos = await oidc.validateAuthenticationProcess(req)
|
2024-04-18 10:23:52 +02:00
|
|
|
logger.debug('external account infos: ' + JSON.stringify(
|
2024-04-17 16:35:26 +02:00
|
|
|
Object.assign(
|
|
|
|
{},
|
|
|
|
externalAccountInfos,
|
|
|
|
{
|
2024-04-17 18:30:39 +02:00
|
|
|
password: '**removed**', // removing the password from logs!
|
2024-04-18 18:25:14 +02:00
|
|
|
token: '**removed**', // same as password
|
|
|
|
avatar: externalAccountInfos.avatar
|
|
|
|
? `**removed** ${externalAccountInfos.avatar.mimetype} avatar`
|
|
|
|
: undefined
|
2024-04-17 16:35:26 +02:00
|
|
|
}
|
|
|
|
)
|
|
|
|
))
|
|
|
|
|
|
|
|
// Now we create or update the user:
|
|
|
|
if (!await ensureUser(options, externalAccountInfos)) {
|
|
|
|
throw new ExternalAuthenticationError(
|
|
|
|
'Failing to create your account, please try again later or report this issue'
|
|
|
|
)
|
|
|
|
}
|
2024-04-16 18:49:23 +02:00
|
|
|
|
2024-04-17 12:09:25 +02:00
|
|
|
res.send(popupResultHTML({
|
|
|
|
ok: true,
|
2024-04-17 18:30:39 +02:00
|
|
|
token: externalAccountInfos.token
|
2024-04-17 12:09:25 +02:00
|
|
|
}))
|
2024-04-16 18:49:23 +02:00
|
|
|
} catch (err) {
|
|
|
|
logger.error('[oidc router] Failed to process the OIDC callback: ' + (err as string))
|
2024-04-17 15:12:37 +02:00
|
|
|
const message = err instanceof ExternalAuthenticationError ? err.message : undefined
|
|
|
|
res.status(500)
|
2024-04-17 12:09:25 +02:00
|
|
|
res.send(popupResultHTML({
|
2024-04-17 15:12:37 +02:00
|
|
|
ok: false,
|
|
|
|
message
|
2024-04-17 12:09:25 +02:00
|
|
|
}))
|
2024-04-16 18:49:23 +02:00
|
|
|
}
|
|
|
|
}
|
2024-04-18 09:52:27 +02:00
|
|
|
)
|
2024-04-22 13:03:31 +02:00
|
|
|
router.get('/:type?/cb', cbHandler)
|
|
|
|
router.post('/:type?/cb', cbHandler)
|
2024-04-16 18:49:23 +02:00
|
|
|
|
|
|
|
return router
|
|
|
|
}
|
|
|
|
|
|
|
|
export {
|
|
|
|
initOIDCRouter
|
|
|
|
}
|