feat-dev: add U2F iframe for desktop client authentication (#2236)

This commit is contained in:
Karol Sójko
2023-03-07 00:52:35 +01:00
committed by GitHub
parent e5ca69fc8e
commit cf5330d7cf
25 changed files with 362 additions and 73 deletions

View File

@@ -4,6 +4,7 @@ import { RefObject, useState } from 'react'
import Button from '../Button/Button'
import Icon from '../Icon/Icon'
import { InputValue } from './InputValue'
import U2FPromptIframeContainer from './U2FPromptIframeContainer'
type Props = {
application: WebApplication
@@ -17,6 +18,18 @@ const U2FPrompt = ({ application, onValueChange, prompt, buttonRef, contextData
const [authenticatorResponse, setAuthenticatorResponse] = useState<Record<string, unknown> | null>(null)
const [error, setError] = useState('')
if (!application.isFullU2FClient) {
return (
<U2FPromptIframeContainer
contextData={contextData}
apiHost={application.getHost() || window.defaultSyncServer}
onResponse={(response) => {
onValueChange(response, prompt)
}}
/>
)
}
return (
<div className="min-w-76">
{error && <div className="text-red-500">{error}</div>}
@@ -27,18 +40,18 @@ const U2FPrompt = ({ application, onValueChange, prompt, buttonRef, contextData
onClick={async () => {
if (!contextData || contextData.username === undefined) {
setError('No username provided')
return
}
const authenticatorResponseOrError = await application.getAuthenticatorAuthenticationResponse.execute({
username: contextData.username,
username: contextData.username as string,
})
if (authenticatorResponseOrError.isFailed()) {
setError(authenticatorResponseOrError.getError())
return
}
const authenticatorResponse = authenticatorResponseOrError.getValue()
setAuthenticatorResponse(authenticatorResponse)

View File

@@ -0,0 +1,66 @@
import { log, LoggingDomain } from '@/Logging'
import { isDev } from '@/Utils'
import { useEffect, useRef } from 'react'
type Props = {
contextData?: Record<string, unknown>
onResponse: (response: string) => void
apiHost: string
}
const U2F_IFRAME_ORIGIN = isDev ? 'http://localhost:3001/?route=u2f' : 'https://app.standardnotes.com/?route=u2f'
const U2FPromptIframeContainer = ({ contextData, onResponse, apiHost }: Props) => {
const iframeRef = useRef<HTMLIFrameElement>(null)
useEffect(() => {
const messageHandler = (event: MessageEvent) => {
log(LoggingDomain.U2F, 'Native client received message', event)
const eventDoesNotComeFromU2FIFrame = event.origin !== new URL(U2F_IFRAME_ORIGIN).origin
if (eventDoesNotComeFromU2FIFrame) {
log(
LoggingDomain.U2F,
'Not sending data to U2F iframe; origin does not match',
event.origin,
new URL(U2F_IFRAME_ORIGIN).origin,
)
return
}
if (event.data.mountedAuthView) {
if (iframeRef.current?.contentWindow) {
log(LoggingDomain.U2F, 'Sending contextData to U2F iframe', contextData)
iframeRef.current.contentWindow.postMessage(
{ username: (contextData as Record<string, unknown>).username, apiHost },
U2F_IFRAME_ORIGIN,
)
}
return
}
if (event.data.assertionResponse) {
log(LoggingDomain.U2F, 'Received assertion response from U2F iframe', event.data.assertionResponse)
onResponse(event.data.assertionResponse)
}
}
window.addEventListener('message', messageHandler)
return () => {
window.removeEventListener('message', messageHandler)
}
}, [contextData, onResponse, apiHost])
return (
<iframe
ref={iframeRef}
src={U2F_IFRAME_ORIGIN}
className="h-40 w-full"
title="U2F"
allow="publickey-credentials-get"
id="u2f"
/>
)
}
export default U2FPromptIframeContainer