Possibility to configure an OpenID Connect provider on the instance level WIP (#128).
This commit is contained in:
parent
43d0fba274
commit
6c75863472
@ -1,4 +1,4 @@
|
|||||||
import type { InitConverseJSParams, ChatIncludeMode } from 'shared/lib/types'
|
import type { InitConverseJSParams, ChatIncludeMode, OIDCAuthResult } from 'shared/lib/types'
|
||||||
import { inIframe } from './lib/utils'
|
import { inIframe } from './lib/utils'
|
||||||
import { initDom } from './lib/dom'
|
import { initDom } from './lib/dom'
|
||||||
import {
|
import {
|
||||||
@ -28,6 +28,7 @@ declare global {
|
|||||||
initConversePlugins: typeof initConversePlugins
|
initConversePlugins: typeof initConversePlugins
|
||||||
initConverse: typeof initConverse
|
initConverse: typeof initConverse
|
||||||
reconnectConverse?: (room: string) => void
|
reconnectConverse?: (room: string) => void
|
||||||
|
oidcGetResult?: (data: OIDCAuthResult) => void
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -6,6 +6,7 @@ import { __ } from 'i18n'
|
|||||||
export default class LivechatExternalLoginContentElement extends CustomElement {
|
export default class LivechatExternalLoginContentElement extends CustomElement {
|
||||||
static get properties () {
|
static get properties () {
|
||||||
return {
|
return {
|
||||||
|
external_auth_oidc_alert_message: { type: String, attribute: false },
|
||||||
remote_peertube_state: { type: String, attribute: false },
|
remote_peertube_state: { type: String, attribute: false },
|
||||||
remote_peertube_alert_message: { type: String, attribute: false },
|
remote_peertube_alert_message: { type: String, attribute: false },
|
||||||
remote_peertube_try_anyway_url: { type: String, attribute: false }
|
remote_peertube_try_anyway_url: { type: String, attribute: false }
|
||||||
@ -19,13 +20,14 @@ export default class LivechatExternalLoginContentElement extends CustomElement {
|
|||||||
|
|
||||||
render () {
|
render () {
|
||||||
return tplExternalLoginModal(this, {
|
return tplExternalLoginModal(this, {
|
||||||
|
external_auth_oidc_alert_message: this.external_auth_oidc_alert_message,
|
||||||
remote_peertube_state: this.remote_peertube_state,
|
remote_peertube_state: this.remote_peertube_state,
|
||||||
remote_peertube_alert_message: this.remote_peertube_alert_message,
|
remote_peertube_alert_message: this.remote_peertube_alert_message,
|
||||||
remote_peertube_try_anyway_url: this.remote_peertube_try_anyway_url
|
remote_peertube_try_anyway_url: this.remote_peertube_try_anyway_url
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
onKeyUp (_ev) {
|
onRemotePeertubeKeyUp (_ev) {
|
||||||
if (this.remote_peertube_state !== 'init') {
|
if (this.remote_peertube_state !== 'init') {
|
||||||
this.remote_peertube_state = 'init'
|
this.remote_peertube_state = 'init'
|
||||||
this.remote_peertube_alert_message = ''
|
this.remote_peertube_alert_message = ''
|
||||||
@ -109,6 +111,7 @@ export default class LivechatExternalLoginContentElement extends CustomElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
clearAlert () {
|
clearAlert () {
|
||||||
|
this.external_auth_oidc_alert_message = ''
|
||||||
this.remote_peertube_alert_message = ''
|
this.remote_peertube_alert_message = ''
|
||||||
this.remote_peertube_try_anyway_url = ''
|
this.remote_peertube_try_anyway_url = ''
|
||||||
}
|
}
|
||||||
|
@ -15,6 +15,16 @@ class ExternalLoginModal extends BaseModal {
|
|||||||
// eslint-disable-next-line no-undef
|
// eslint-disable-next-line no-undef
|
||||||
return __(LOC_login_using_external_account)
|
return __(LOC_login_using_external_account)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
onHide () {
|
||||||
|
super.onHide()
|
||||||
|
// kill the oidcGetResult handler if still there
|
||||||
|
try {
|
||||||
|
if (window.oidcGetResult) { window.oidcGetResult() }
|
||||||
|
} catch (err) {
|
||||||
|
console.error(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
api.elements.define('converse-livechat-external-login', ExternalLoginModal)
|
api.elements.define('converse-livechat-external-login', ExternalLoginModal)
|
||||||
|
@ -17,10 +17,48 @@ export const tplExternalLoginModal = (el, o) => {
|
|||||||
<div class="livechat-external-login-modal-external-auth-oidc">
|
<div class="livechat-external-login-modal-external-auth-oidc">
|
||||||
<button
|
<button
|
||||||
class="btn btn-primary"
|
class="btn btn-primary"
|
||||||
@click=${() => window.open(externalAuthOIDCUrl)}
|
@click=${
|
||||||
|
(ev) => {
|
||||||
|
ev.preventDefault()
|
||||||
|
|
||||||
|
el.clearAlert()
|
||||||
|
|
||||||
|
const popup = window.open(
|
||||||
|
externalAuthOIDCUrl,
|
||||||
|
'livechat-oidc',
|
||||||
|
'popup'
|
||||||
|
)
|
||||||
|
|
||||||
|
window.oidcGetResult = (data) => {
|
||||||
|
window.oidcGetResult = undefined
|
||||||
|
|
||||||
|
if (!data) {
|
||||||
|
// special case: when this modal is closed, used to close the popup
|
||||||
|
if (popup) { popup.close() }
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log('Received an OIDC authentication result...', data)
|
||||||
|
if (!data.ok) {
|
||||||
|
// eslint-disable-next-line no-undef
|
||||||
|
el.external_auth_oidc_alert_message = __(LOC_login_external_oidc_alert_message) +
|
||||||
|
(data.message ? ` (${data.message})` : '')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// TODO
|
||||||
|
console.error('not implemented yet')
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
>
|
>
|
||||||
${externalAuthOIDCButtonLabel}
|
${externalAuthOIDCButtonLabel}
|
||||||
</button>
|
</button>
|
||||||
|
${!o.external_auth_oidc_alert_message
|
||||||
|
? ''
|
||||||
|
: html`<div class="invalid-feedback d-block">${o.external_auth_oidc_alert_message}</div>`
|
||||||
|
}
|
||||||
</div>
|
</div>
|
||||||
<hr>
|
<hr>
|
||||||
`
|
`
|
||||||
@ -33,7 +71,7 @@ export const tplExternalLoginModal = (el, o) => {
|
|||||||
placeholder="${i18nRemotePeertubeUrl}"
|
placeholder="${i18nRemotePeertubeUrl}"
|
||||||
class="form-control ${o.remote_peertube_alert_message ? 'is-invalid' : ''}"
|
class="form-control ${o.remote_peertube_alert_message ? 'is-invalid' : ''}"
|
||||||
name="peertube_url"
|
name="peertube_url"
|
||||||
@keyup=${el.onKeyUp}
|
@keyup=${el.onRemotePeertubeKeyUp}
|
||||||
?disabled=${o.remote_peertube_state === 'loading'}
|
?disabled=${o.remote_peertube_state === 'loading'}
|
||||||
/>
|
/>
|
||||||
</label>
|
</label>
|
||||||
|
@ -13,7 +13,8 @@ const locKeys = [
|
|||||||
'login_remote_peertube_no_livechat',
|
'login_remote_peertube_no_livechat',
|
||||||
'login_remote_peertube_video_not_found',
|
'login_remote_peertube_video_not_found',
|
||||||
'login_remote_peertube_video_not_found_try_anyway',
|
'login_remote_peertube_video_not_found_try_anyway',
|
||||||
'login_remote_peertube_video_not_found_try_anyway_button'
|
'login_remote_peertube_video_not_found_try_anyway_button',
|
||||||
|
'login_external_oidc_alert_message'
|
||||||
]
|
]
|
||||||
|
|
||||||
module.exports = locKeys
|
module.exports = locKeys
|
||||||
|
@ -421,3 +421,4 @@ login_remote_peertube_no_livechat: "The livechat plugin is not installed on this
|
|||||||
login_remote_peertube_video_not_found: "This video is not available on this Peertube instance."
|
login_remote_peertube_video_not_found: "This video is not available on this Peertube instance."
|
||||||
login_remote_peertube_video_not_found_try_anyway: "In some cases, the video can still be retrieved if you connect to the remote instance."
|
login_remote_peertube_video_not_found_try_anyway: "In some cases, the video can still be retrieved if you connect to the remote instance."
|
||||||
login_remote_peertube_video_not_found_try_anyway_button: "Try anyway to open the video on the Peertube instance"
|
login_remote_peertube_video_not_found_try_anyway_button: "Try anyway to open the video on the Peertube instance"
|
||||||
|
login_external_oidc_alert_message: "Authentication failed"
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import type { RegisterServerOptions } from '@peertube/peertube-types'
|
import type { RegisterServerOptions } from '@peertube/peertube-types'
|
||||||
import type { Request } from 'express'
|
import type { Request, Response, CookieOptions } from 'express'
|
||||||
import { URL } from 'url'
|
import { URL } from 'url'
|
||||||
import { Issuer, BaseClient, generators } from 'openid-client'
|
import { Issuer, BaseClient, generators } from 'openid-client'
|
||||||
import { getBaseRouterRoute } from '../helpers'
|
import { getBaseRouterRoute } from '../helpers'
|
||||||
@ -37,6 +37,14 @@ class ExternalAuthOIDC {
|
|||||||
outputEncoding: 'hex' as Encoding
|
outputEncoding: 'hex' as Encoding
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private readonly cookieNamePrefix: string = 'peertube-plugin-livechat-oidc-'
|
||||||
|
private readonly cookieOptions: CookieOptions = {
|
||||||
|
secure: true,
|
||||||
|
httpOnly: true,
|
||||||
|
sameSite: 'none',
|
||||||
|
maxAge: 1000 * 60 * 10 // 10 minutes
|
||||||
|
}
|
||||||
|
|
||||||
private ok: boolean | undefined
|
private ok: boolean | undefined
|
||||||
|
|
||||||
private issuer: Issuer | undefined | null
|
private issuer: Issuer | undefined | null
|
||||||
@ -217,12 +225,11 @@ class ExternalAuthOIDC {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns everything that is needed to instanciate an OIDC authentication.
|
* Returns everything that is needed to instanciate an OIDC authentication.
|
||||||
|
* @param req express request
|
||||||
|
* @param res express response. Will add some cookies.
|
||||||
|
* @return the url to which redirect
|
||||||
*/
|
*/
|
||||||
async initAuthenticationProcess (): Promise<{
|
async initAuthenticationProcess (req: Request, res: Response): Promise<string> {
|
||||||
encryptedCodeVerifier: string
|
|
||||||
encryptedState: string
|
|
||||||
redirectUrl: string
|
|
||||||
}> {
|
|
||||||
if (!this.client) {
|
if (!this.client) {
|
||||||
throw new Error('External Auth OIDC not loaded yet, too soon to call oidc.initAuthentication')
|
throw new Error('External Auth OIDC not loaded yet, too soon to call oidc.initAuthentication')
|
||||||
}
|
}
|
||||||
@ -242,29 +249,27 @@ class ExternalAuthOIDC {
|
|||||||
state
|
state
|
||||||
})
|
})
|
||||||
|
|
||||||
return {
|
res.cookie(this.cookieNamePrefix + 'code-verifier', encryptedCodeVerifier, this.cookieOptions)
|
||||||
encryptedCodeVerifier,
|
res.cookie(this.cookieNamePrefix + 'state', encryptedState, this.cookieOptions)
|
||||||
encryptedState,
|
return redirectUrl
|
||||||
redirectUrl
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Authentication process callback.
|
* Authentication process callback.
|
||||||
* @param req The ExpressJS request object.
|
* @param req The ExpressJS request object. Will read cookies.
|
||||||
* @return user info
|
* @return user info
|
||||||
*/
|
*/
|
||||||
async validateAuthenticationProcess (req: Request, cookieNamePrefix: string): Promise<any> {
|
async validateAuthenticationProcess (req: Request): Promise<any> {
|
||||||
if (!this.client) {
|
if (!this.client) {
|
||||||
throw new Error('External Auth OIDC not loaded yet, too soon to call oidc.validateAuthenticationProcess')
|
throw new Error('External Auth OIDC not loaded yet, too soon to call oidc.validateAuthenticationProcess')
|
||||||
}
|
}
|
||||||
|
|
||||||
const encryptedCodeVerifier = req.cookies[cookieNamePrefix + 'code-verifier']
|
const encryptedCodeVerifier = req.cookies[this.cookieNamePrefix + 'code-verifier']
|
||||||
if (!encryptedCodeVerifier) {
|
if (!encryptedCodeVerifier) {
|
||||||
throw new Error('Received callback but code verifier not found in request cookies.')
|
throw new Error('Received callback but code verifier not found in request cookies.')
|
||||||
}
|
}
|
||||||
|
|
||||||
const encryptedState = req.cookies[cookieNamePrefix + 'state']
|
const encryptedState = req.cookies[this.cookieNamePrefix + 'state']
|
||||||
if (!encryptedState) {
|
if (!encryptedState) {
|
||||||
throw new Error('Received callback but state not found in request cookies.')
|
throw new Error('Received callback but state not found in request cookies.')
|
||||||
}
|
}
|
||||||
|
@ -1,14 +1,32 @@
|
|||||||
import type { RegisterServerOptions } from '@peertube/peertube-types'
|
import type { RegisterServerOptions } from '@peertube/peertube-types'
|
||||||
import type { Router, Request, Response, NextFunction, CookieOptions } from 'express'
|
import type { Router, Request, Response, NextFunction } from 'express'
|
||||||
|
import type { OIDCAuthResult } from '../../../shared/lib/types'
|
||||||
import { asyncMiddleware } from '../middlewares/async'
|
import { asyncMiddleware } from '../middlewares/async'
|
||||||
import { ExternalAuthOIDC } from '../external-auth/oidc'
|
import { ExternalAuthOIDC } from '../external-auth/oidc'
|
||||||
|
|
||||||
const cookieNamePrefix = 'peertube-plugin-livechat-oidc-'
|
/**
|
||||||
const cookieOptions: CookieOptions = {
|
* When using a popup for OIDC, writes the HTML/Javascript to close the popup
|
||||||
secure: true,
|
* and send the result to the parent window.
|
||||||
httpOnly: true,
|
* @param result the result to send to the parent window
|
||||||
sameSite: 'none',
|
*/
|
||||||
maxAge: 1000 * 60 * 10 // 10 minutes
|
function popupResultHTML (result: OIDCAuthResult): string {
|
||||||
|
return `<!DOCTYPE html><html>
|
||||||
|
<body>
|
||||||
|
<noscript>Your browser must enable javascript for this page to work.</noscript>
|
||||||
|
<script>
|
||||||
|
try {
|
||||||
|
const data = ${JSON.stringify(result)};
|
||||||
|
if (!window.opener || !window.opener.oidcGetResult) {
|
||||||
|
throw new Error("Can't find parent window callback handler.")
|
||||||
|
}
|
||||||
|
window.opener.oidcGetResult(data);
|
||||||
|
window.close();
|
||||||
|
} catch (err) {
|
||||||
|
document.body.innerText = 'Error: ' + err;
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html> `
|
||||||
}
|
}
|
||||||
|
|
||||||
async function initOIDCRouter (options: RegisterServerOptions): Promise<Router> {
|
async function initOIDCRouter (options: RegisterServerOptions): Promise<Router> {
|
||||||
@ -26,10 +44,8 @@ async function initOIDCRouter (options: RegisterServerOptions): Promise<Router>
|
|||||||
throw new Error('[oidc router] External Auth OIDC not loaded yet')
|
throw new Error('[oidc router] External Auth OIDC not loaded yet')
|
||||||
}
|
}
|
||||||
|
|
||||||
const authenticationProcess = await oidc.initAuthenticationProcess()
|
const redirectUrl = await oidc.initAuthenticationProcess(req, res)
|
||||||
res.cookie(cookieNamePrefix + 'code-verifier', authenticationProcess.encryptedCodeVerifier, cookieOptions)
|
res.redirect(redirectUrl)
|
||||||
res.cookie(cookieNamePrefix + 'state', authenticationProcess.encryptedState, cookieOptions)
|
|
||||||
return res.redirect(authenticationProcess.redirectUrl)
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
logger.error('[oidc router] Failed to process the OIDC callback: ' + (err as string))
|
logger.error('[oidc router] Failed to process the OIDC callback: ' + (err as string))
|
||||||
next()
|
next()
|
||||||
@ -38,7 +54,7 @@ async function initOIDCRouter (options: RegisterServerOptions): Promise<Router>
|
|||||||
))
|
))
|
||||||
|
|
||||||
router.get('/cb', asyncMiddleware(
|
router.get('/cb', asyncMiddleware(
|
||||||
async (req: Request, res: Response, next: NextFunction) => {
|
async (req: Request, res: Response, _next: NextFunction) => {
|
||||||
logger.info('[oidc router] OIDC callback call')
|
logger.info('[oidc router] OIDC callback call')
|
||||||
try {
|
try {
|
||||||
const oidc = ExternalAuthOIDC.singleton()
|
const oidc = ExternalAuthOIDC.singleton()
|
||||||
@ -47,13 +63,20 @@ async function initOIDCRouter (options: RegisterServerOptions): Promise<Router>
|
|||||||
throw new Error('[oidc router] External Auth OIDC not loaded yet')
|
throw new Error('[oidc router] External Auth OIDC not loaded yet')
|
||||||
}
|
}
|
||||||
|
|
||||||
const userInfos = await oidc.validateAuthenticationProcess(req, cookieNamePrefix)
|
const userInfos = await oidc.validateAuthenticationProcess(req)
|
||||||
logger.info(JSON.stringify(userInfos)) // FIXME
|
logger.info(JSON.stringify(userInfos)) // FIXME (normalize data type, process, ...)
|
||||||
|
|
||||||
res.send('ok')
|
res.send(popupResultHTML({
|
||||||
|
ok: true,
|
||||||
|
username: userInfos.username,
|
||||||
|
password: 'TODO'
|
||||||
|
}))
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
logger.error('[oidc router] Failed to process the OIDC callback: ' + (err as string))
|
logger.error('[oidc router] Failed to process the OIDC callback: ' + (err as string))
|
||||||
next()
|
res.sendStatus(500)
|
||||||
|
res.send(popupResultHTML({
|
||||||
|
ok: false
|
||||||
|
}))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
))
|
))
|
||||||
|
@ -107,6 +107,19 @@ type ChatPeertubeIncludeMode = 'peertube-fullpage' | 'peertube-video'
|
|||||||
*/
|
*/
|
||||||
type ChatIncludeMode = 'chat-only' | ChatPeertubeIncludeMode
|
type ChatIncludeMode = 'chat-only' | ChatPeertubeIncludeMode
|
||||||
|
|
||||||
|
interface OIDCAuthResultError {
|
||||||
|
ok: true
|
||||||
|
username: string
|
||||||
|
password: string
|
||||||
|
}
|
||||||
|
|
||||||
|
interface OIDCAuthResultOk {
|
||||||
|
ok: false
|
||||||
|
message?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
type OIDCAuthResult = OIDCAuthResultError | OIDCAuthResultOk
|
||||||
|
|
||||||
export type {
|
export type {
|
||||||
ConverseJSTheme,
|
ConverseJSTheme,
|
||||||
InitConverseJSParams,
|
InitConverseJSParams,
|
||||||
@ -117,5 +130,8 @@ export type {
|
|||||||
ChannelConfigurationOptions,
|
ChannelConfigurationOptions,
|
||||||
ChannelConfiguration,
|
ChannelConfiguration,
|
||||||
ChatIncludeMode,
|
ChatIncludeMode,
|
||||||
ChatPeertubeIncludeMode
|
ChatPeertubeIncludeMode,
|
||||||
|
OIDCAuthResultError,
|
||||||
|
OIDCAuthResultOk,
|
||||||
|
OIDCAuthResult
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user