Adding some standard OpenID Connect providers (Google, Facebook) (WIP):

* frontend
This commit is contained in:
John Livingston 2024-04-22 14:28:55 +02:00
parent 024186ba2c
commit 0a492d1921
No known key found for this signature in database
GPG Key ID: B17B5640CE66CDBC
8 changed files with 148 additions and 93 deletions

View File

@ -109,7 +109,7 @@ async function initConverse (
let isRemoteWithNicknameSet: boolean = false let isRemoteWithNicknameSet: boolean = false
// OIDC (OpenID Connect): // OIDC (OpenID Connect):
const tryOIDC = !!initConverseParams.externalAuthOIDC const tryOIDC = (initConverseParams.externalAuthOIDC?.length ?? 0) > 0
const auth = await getLocalAuthentInfos(authenticationUrl, tryOIDC, peertubeAuthHeader) const auth = await getLocalAuthentInfos(authenticationUrl, tryOIDC, peertubeAuthHeader)
@ -171,8 +171,7 @@ async function initConverse (
params.livechat_specific_external_authent = isAuthenticatedWithExternalAccount params.livechat_specific_external_authent = isAuthenticatedWithExternalAccount
if (tryOIDC && !isAuthenticated) { if (tryOIDC && !isAuthenticated) {
params.livechat_external_auth_oidc_button_label = initConverseParams.externalAuthOIDC?.buttonLabel params.livechat_external_auth_oidc_buttons = initConverseParams.externalAuthOIDC
params.livechat_external_auth_oidc_url = initConverseParams.externalAuthOIDC?.url
} }
if (tryOIDC) { // also needed when authenticated (for the signout button) if (tryOIDC) { // also needed when authenticated (for the signout button)

View File

@ -94,8 +94,13 @@ body[livechat-viewer-mode="on"] {
} }
.livechat-external-login-modal { .livechat-external-login-modal {
.livechat-external-login-modal-external-auth-oidc { .livechat-external-login-modal-external-auth-oidc-block {
text-align: center; justify-content: center;
align-items: center;
display: flex;
flex-flow: row wrap;
gap: 20px;
margin-bottom: 20px;
width: 100%; width: 100%;
} }

View File

@ -2,23 +2,7 @@ import { _converse, api } from '@converse/headless/core'
import { __ } from 'i18n' import { __ } from 'i18n'
import { html } from 'lit' import { html } from 'lit'
export const tplExternalLoginModal = (el, o) => { function externalLoginClickHandler (ev, el, externalAuthOIDCUrl) {
// eslint-disable-next-line no-undef
const i18nRemotePeertube = __(LOC_login_remote_peertube)
// eslint-disable-next-line no-undef
const i18nRemotePeertubeUrl = __(LOC_login_remote_peertube_url)
const i18nRemotePeertubeOpen = __('OK')
const externalAuthOIDCButtonLabel = api.settings.get('livechat_external_auth_oidc_button_label')
const externalAuthOIDCUrl = api.settings.get('livechat_external_auth_oidc_url')
return html`<div class="modal-body livechat-external-login-modal">
${!externalAuthOIDCButtonLabel || !externalAuthOIDCUrl || !window.sessionStorage
? ''
: html`
<div class="livechat-external-login-modal-external-auth-oidc">
<button
class="btn btn-primary"
@click=${
(ev) => {
ev.preventDefault() ev.preventDefault()
el.clearAlert() el.clearAlert()
@ -80,10 +64,71 @@ export const tplExternalLoginModal = (el, o) => {
return false return false
} }
export const tplExternalLoginModal = (el, o) => {
// eslint-disable-next-line no-undef
const i18nRemotePeertube = __(LOC_login_remote_peertube)
// eslint-disable-next-line no-undef
const i18nRemotePeertubeUrl = __(LOC_login_remote_peertube_url)
const i18nRemotePeertubeOpen = __('OK')
const buttonsDefinitions = api.settings.get('livechat_external_auth_oidc_buttons')
const externalButtonsBlocks = []
if (Array.isArray(buttonsDefinitions) && buttonsDefinitions.length) {
// type=custom first, if present
// and sorting on label alphabetically
const customButtonsDefinitions = buttonsDefinitions.filter(b => b.type === 'custom')
.sort((a, b) => a.buttonLabel > b.buttonLabel ? 1 : (a.buttonLabel < b.buttonLabel ? -1 : 0))
const otherButtonsDefinition = buttonsDefinitions.filter(b => b.type !== 'custom')
.sort((a, b) => a.buttonLabel > b.buttonLabel ? 1 : (a.buttonLabel < b.buttonLabel ? -1 : 0))
for (const block of [customButtonsDefinitions, otherButtonsDefinition]) {
if (!block.length) { continue }
const externalButtons = []
for (const buttonDef of block) {
if (typeof buttonDef !== 'object') { continue }
const type = buttonDef.type
const label = buttonDef.buttonLabel
const url = buttonDef.url
if (!type || !type || !url) {
continue
}
externalButtons.push({
label,
url,
class: 'livechat-external-login-modal-external-auth-oidc-type-' + type
})
}
externalButtonsBlocks.push(externalButtons)
}
}
return html`<div class="modal-body livechat-external-login-modal">
${!externalButtonsBlocks.length || !window.sessionStorage
? ''
: html`<div class="livechat-external-login-modal-external-auth-oidc">
${
externalButtonsBlocks.map(externalButtons => html`
<div class="livechat-external-login-modal-external-auth-oidc-block">
${
externalButtons.map(button => html`
<button
class="btn btn-primary ${button.class}"
@click=${
(ev) => {
externalLoginClickHandler(ev, el, button.url)
}
} }
> >
${externalAuthOIDCButtonLabel} ${button.label}
</button> </button>
`)
}
</div>
`)
}
${!o.external_auth_oidc_alert_message ${!o.external_auth_oidc_alert_message
? '' ? ''
: html`<div class="invalid-feedback d-block">${o.external_auth_oidc_alert_message}</div>` : html`<div class="invalid-feedback d-block">${o.external_auth_oidc_alert_message}</div>`

View File

@ -82,7 +82,7 @@ export const livechatSpecificsPlugin = {
for (const k of [ for (const k of [
'hide_muc_participants', 'hide_muc_participants',
'livechat_enable_viewer_mode', 'livechat_enable_viewer_mode',
'livechat_external_auth_oidc_button_label', 'livechat_external_auth_oidc_url', 'livechat_external_auth_oidc_buttons',
'livechat_external_auth_reconnect_mode', 'livechat_external_auth_reconnect_mode',
'livechat_mini_muc_head', 'livechat_mini_muc_head',
'livechat_specific_external_authent' 'livechat_specific_external_authent'

View File

@ -9,8 +9,7 @@ export const livechatViewerModePlugin = {
livechat_enable_viewer_mode: false, livechat_enable_viewer_mode: false,
livechat_peertube_video_original_url: undefined, livechat_peertube_video_original_url: undefined,
livechat_peertube_video_uuid: undefined, livechat_peertube_video_uuid: undefined,
livechat_external_auth_oidc_button_label: undefined, livechat_external_auth_oidc_buttons: undefined,
livechat_external_auth_oidc_url: undefined,
livechat_external_auth_reconnect_mode: undefined livechat_external_auth_reconnect_mode: undefined
}) })

View File

@ -85,14 +85,18 @@ async function getConverseJSParams (
) )
} else { } else {
try { try {
const customOidc = ExternalAuthOIDC.singleton('custom') const oidcs = ExternalAuthOIDC.allSingletons()
if (await customOidc.isOk()) { for (const oidc of oidcs) {
const authUrl = customOidc.getConnectUrl() if (await oidc.isOk()) {
const buttonLabel = customOidc.getButtonLabel() const authUrl = oidc.getConnectUrl()
const buttonLabel = oidc.getButtonLabel()
if (authUrl && buttonLabel) { if (authUrl && buttonLabel) {
externalAuthOIDC = { externalAuthOIDC ??= []
externalAuthOIDC.push({
type: oidc.type,
buttonLabel: buttonLabel, buttonLabel: buttonLabel,
url: authUrl url: authUrl
})
} }
} }
} }

View File

@ -1,6 +1,7 @@
import type { RegisterServerOptions } from '@peertube/peertube-types' import type { RegisterServerOptions } from '@peertube/peertube-types'
import type { Request, Response, CookieOptions } from 'express' import type { Request, Response, CookieOptions } from 'express'
import type { ExternalAccountInfos, AcceptableAvatarMimeType } from './types' import type { ExternalAccountInfos, AcceptableAvatarMimeType } from './types'
import type { ExternalAuthOIDCType } from '../../../shared/lib/types'
import { ExternalAuthenticationError } from './error' import { ExternalAuthenticationError } from './error'
import { getBaseRouterRoute } from '../helpers' import { getBaseRouterRoute } from '../helpers'
import { canonicalizePluginUri } from '../uri/canonicalize' import { canonicalizePluginUri } from '../uri/canonicalize'
@ -68,8 +69,6 @@ async function getRandomBytes (size: number): Promise<Buffer> {
}) })
} }
type ExternalAuthOIDCType = 'custom' | 'google' | 'facebook'
/** /**
* This class handles the external OpenId Connect provider, if defined. * This class handles the external OpenId Connect provider, if defined.
*/ */

View File

@ -22,10 +22,11 @@ interface InitConverseJSParams {
transparent: boolean transparent: boolean
forceDefaultHideMucParticipants?: boolean forceDefaultHideMucParticipants?: boolean
autofocus?: boolean autofocus?: boolean
externalAuthOIDC?: { externalAuthOIDC?: Array<{
type: ExternalAuthOIDCType
buttonLabel: string buttonLabel: string
url: string url: string
} }>
} }
interface InitConverseJSParamsError { interface InitConverseJSParamsError {
@ -119,6 +120,8 @@ interface ExternalAuthResultError {
type ExternalAuthResult = ExternalAuthResultError | ExternalAuthResultOk type ExternalAuthResult = ExternalAuthResultError | ExternalAuthResultOk
type ExternalAuthOIDCType = 'custom' | 'google' | 'facebook'
export type { export type {
ConverseJSTheme, ConverseJSTheme,
InitConverseJSParams, InitConverseJSParams,
@ -132,5 +135,6 @@ export type {
ChatPeertubeIncludeMode, ChatPeertubeIncludeMode,
ExternalAuthResultError, ExternalAuthResultError,
ExternalAuthResultOk, ExternalAuthResultOk,
ExternalAuthResult ExternalAuthResult,
ExternalAuthOIDCType
} }