2023-07-03 02:39:23 -06:00
|
|
|
/*
|
|
|
|
Copyright 2023 Yarmo Mackenbach
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
2023-07-08 00:17:13 -06:00
|
|
|
import axios from 'axios'
|
|
|
|
import { decodeProtectedHeader, importJWK, compactVerify, calculateJwkThumbprint } from 'jose'
|
|
|
|
import { base32, base64url } from 'rfc4648'
|
|
|
|
import { Claim } from './claim.js'
|
|
|
|
import { Persona } from './persona.js'
|
|
|
|
import { Profile } from './profile.js'
|
2023-07-13 02:39:13 -06:00
|
|
|
import { ProfileType, PublicKeyEncoding, PublicKeyFetchMethod, PublicKeyType } from './enums.js'
|
2023-07-03 02:39:23 -06:00
|
|
|
|
|
|
|
const SupportedCryptoAlg = ['EdDSA', 'ES256', 'ES256K', 'ES384', 'ES512']
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Functions related to Ariadne Signature Profiles
|
|
|
|
* @module asp
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fetch a public key using Web Key Directory
|
|
|
|
* @function
|
|
|
|
* @param {string} uri - ASPE URI
|
|
|
|
* @returns {Promise<Profile>}
|
|
|
|
* @example
|
|
|
|
* const key = doip.aspe.fetchASPE('aspe:domain.tld:1234567890');
|
|
|
|
*/
|
2023-07-08 00:17:13 -06:00
|
|
|
export async function fetchASPE (uri) {
|
2023-07-03 02:39:23 -06:00
|
|
|
const re = /aspe:(.*):(.*)/
|
|
|
|
|
|
|
|
if (!re.test(uri)) {
|
|
|
|
throw new Error('Invalid ASPE URI')
|
|
|
|
}
|
|
|
|
|
|
|
|
const matches = uri.match(re)
|
|
|
|
const domainPart = matches[1]
|
|
|
|
const localPart = matches[2].toUpperCase()
|
|
|
|
|
|
|
|
const profileUrl = `https://${domainPart}/.well-known/aspe/id/${localPart}`
|
|
|
|
let profileJws
|
|
|
|
|
|
|
|
try {
|
|
|
|
profileJws = await axios.get(
|
|
|
|
profileUrl,
|
|
|
|
{
|
|
|
|
responseType: 'text'
|
|
|
|
}
|
|
|
|
)
|
|
|
|
.then((/** @type {import('axios').AxiosResponse} */ response) => {
|
|
|
|
if (response.status === 200) {
|
|
|
|
return response
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.then((/** @type {import('axios').AxiosResponse} */ response) => response.data)
|
|
|
|
} catch (e) {
|
|
|
|
throw new Error(`Error fetching Keybase key: ${e.message}`)
|
|
|
|
}
|
|
|
|
|
2023-07-13 02:39:13 -06:00
|
|
|
const profile = await parseProfileJws(profileJws, uri)
|
|
|
|
profile.publicKey.fetch.method = PublicKeyFetchMethod.ASPE
|
|
|
|
profile.publicKey.fetch.query = uri
|
|
|
|
profile.publicKey.fetch.resolvedUrl = profileUrl
|
|
|
|
|
|
|
|
return profile
|
2023-07-03 02:39:23 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fetch a public key using Web Key Directory
|
|
|
|
* @function
|
|
|
|
* @param {string} profileJws - Compact-Serialized profile JWS
|
|
|
|
* @param {string} uri - The ASPE URI associated with the profile
|
|
|
|
* @returns {Promise<Profile>}
|
|
|
|
* @example
|
|
|
|
* const key = doip.aspe.parseProfileJws('...');
|
|
|
|
*/
|
2023-07-08 00:17:13 -06:00
|
|
|
export async function parseProfileJws (profileJws, uri) {
|
2023-07-03 02:39:23 -06:00
|
|
|
const matches = uri.match(/aspe:(.*):(.*)/)
|
|
|
|
const localPart = matches[2].toUpperCase()
|
|
|
|
|
|
|
|
// Decode the headers
|
2023-07-08 00:17:13 -06:00
|
|
|
const protectedHeader = decodeProtectedHeader(profileJws)
|
2023-07-03 02:39:23 -06:00
|
|
|
|
|
|
|
// Extract the JWK
|
|
|
|
if (!SupportedCryptoAlg.includes(protectedHeader.alg)) {
|
|
|
|
throw new Error('Invalid profile JWS: wrong key algorithm')
|
|
|
|
}
|
|
|
|
if (!protectedHeader.kid) {
|
|
|
|
throw new Error('Invalid profile JWS: missing key identifier')
|
|
|
|
}
|
|
|
|
if (!protectedHeader.jwk) {
|
|
|
|
throw new Error('Invalid profile JWS: missing key')
|
|
|
|
}
|
2023-07-08 00:17:13 -06:00
|
|
|
const publicKey = await importJWK(protectedHeader.jwk, protectedHeader.alg)
|
2023-07-03 02:39:23 -06:00
|
|
|
|
|
|
|
// Compute and verify the fingerprint
|
|
|
|
const fp = await computeJwkFingerprint(protectedHeader.jwk)
|
|
|
|
|
|
|
|
if (fp !== protectedHeader.kid) {
|
|
|
|
throw new Error('Invalid profile JWS: wrong key')
|
|
|
|
}
|
|
|
|
if (localPart && fp !== localPart) {
|
|
|
|
throw new Error('Invalid profile JWS: wrong key')
|
|
|
|
}
|
|
|
|
|
|
|
|
// Decode the payload
|
2023-07-08 00:17:13 -06:00
|
|
|
const { payload } = await compactVerify(profileJws, publicKey)
|
2023-07-03 02:39:23 -06:00
|
|
|
const payloadJson = JSON.parse(new TextDecoder().decode(payload))
|
|
|
|
|
|
|
|
// Verify the payload
|
|
|
|
if (!(Object.prototype.hasOwnProperty.call(payloadJson, 'http://ariadne.id/type') && payloadJson['http://ariadne.id/type'] === 'profile')) {
|
|
|
|
throw new Error('Invalid profile JWS: JWS is not a profile')
|
|
|
|
}
|
|
|
|
if (!(Object.prototype.hasOwnProperty.call(payloadJson, 'http://ariadne.id/version') && payloadJson['http://ariadne.id/version'] === 0)) {
|
|
|
|
throw new Error('Invalid profile JWS: profile version not supported')
|
|
|
|
}
|
|
|
|
|
|
|
|
// Extract data from the payload
|
|
|
|
/** @type {string} */
|
|
|
|
const profileName = payloadJson['http://ariadne.id/name']
|
|
|
|
/** @type {string} */
|
|
|
|
const profileDescription = payloadJson['http://ariadne.id/description']
|
|
|
|
/** @type {string[]} */
|
|
|
|
const profileClaims = payloadJson['http://ariadne.id/claims']
|
|
|
|
|
|
|
|
const profileClaimsParsed = profileClaims.map(x => new Claim(x, uri))
|
|
|
|
|
2023-07-09 04:03:25 -06:00
|
|
|
const pe = new Persona(profileName, profileClaimsParsed)
|
|
|
|
if (profileDescription) {
|
|
|
|
pe.setDescription(profileDescription)
|
|
|
|
}
|
|
|
|
|
2023-07-13 02:39:13 -06:00
|
|
|
const profile = new Profile(ProfileType.ASP, uri, [pe])
|
|
|
|
profile.publicKey.fingerprint = fp
|
|
|
|
profile.publicKey.encoding = PublicKeyEncoding.JWK
|
|
|
|
profile.publicKey.encodedKey = JSON.stringify(protectedHeader.jwk)
|
|
|
|
profile.publicKey.key = protectedHeader.jwk
|
|
|
|
|
|
|
|
switch (protectedHeader.alg) {
|
|
|
|
case 'ES256':
|
|
|
|
profile.publicKey.keyType = PublicKeyType.ES256
|
|
|
|
break
|
|
|
|
|
|
|
|
case 'EdDSA':
|
|
|
|
profile.publicKey.keyType = PublicKeyType.EDDSA
|
|
|
|
break
|
|
|
|
|
|
|
|
default:
|
|
|
|
profile.publicKey.keyType = PublicKeyType.UNKNOWN
|
|
|
|
break
|
|
|
|
}
|
2023-07-03 02:39:23 -06:00
|
|
|
|
2023-07-13 02:39:13 -06:00
|
|
|
return profile
|
2023-07-03 02:39:23 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Compute the fingerprint for JWK keys
|
|
|
|
* @function
|
2023-07-08 00:17:13 -06:00
|
|
|
* @param {import('jose').JWK} key
|
2023-07-03 02:39:23 -06:00
|
|
|
* @returns {Promise<string>}
|
|
|
|
*/
|
2023-07-08 00:17:13 -06:00
|
|
|
export async function computeJwkFingerprint (key) {
|
|
|
|
const thumbprint = await calculateJwkThumbprint(key, 'sha512')
|
2023-07-03 02:39:23 -06:00
|
|
|
const fingerprintBytes = base64url.parse(thumbprint, { loose: true }).slice(0, 16)
|
|
|
|
const fingerprint = base32.stringify(fingerprintBytes, { pad: false })
|
|
|
|
|
|
|
|
return fingerprint
|
|
|
|
}
|