mirror of
https://codeberg.org/keyoxide/doipjs.git
synced 2024-12-22 14:39:28 -07:00
feat: support ASPE claims
This commit is contained in:
parent
cd96131ad1
commit
4a77591c57
6 changed files with 169 additions and 0 deletions
|
@ -40,6 +40,8 @@ export const ProxyPolicy = {
|
|||
export const Fetcher = {
|
||||
/** HTTP requests to ActivityPub */
|
||||
ACTIVITYPUB: 'activitypub',
|
||||
/** ASPE HTTP requests */
|
||||
ASPE: 'aspe',
|
||||
/** DNS module from Node.js */
|
||||
DNS: 'dns',
|
||||
/** GraphQL over HTTP requests */
|
||||
|
|
77
src/fetcher/aspe.js
Normal file
77
src/fetcher/aspe.js
Normal file
|
@ -0,0 +1,77 @@
|
|||
/*
|
||||
Copyright 2024 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.
|
||||
*/
|
||||
import axios from 'axios'
|
||||
import isFQDN from 'validator/lib/isFQDN.js'
|
||||
import { version } from '../constants.js'
|
||||
import { parseProfileJws } from '../asp.js'
|
||||
|
||||
export const timeout = 5000
|
||||
|
||||
/**
|
||||
* Execute a fetch request
|
||||
* @function
|
||||
* @async
|
||||
* @param {object} data - Data used in the request
|
||||
* @param {string} data.aspeUri - ASPE URI of the targeted profile
|
||||
* @param {number} [data.fetcherTimeout] - Optional timeout for the fetcher
|
||||
* @returns {Promise<object|string>}
|
||||
*/
|
||||
export async function fn (data, opts) {
|
||||
let timeoutHandle
|
||||
const timeoutPromise = new Promise((resolve, reject) => {
|
||||
timeoutHandle = setTimeout(
|
||||
() => reject(new Error('Request was timed out')),
|
||||
data.fetcherTimeout ? data.fetcherTimeout : timeout
|
||||
)
|
||||
})
|
||||
|
||||
const fetchPromise = new Promise((resolve, reject) => {
|
||||
const reURI = /^aspe:([a-zA-Z0-9.\-_]*):([a-zA-Z0-9]*)/
|
||||
const match = data.aspeUri.match(reURI)
|
||||
|
||||
if (!data.aspeUri || !reURI.test(data.aspeUri) || !isFQDN(match[1])) {
|
||||
reject(new Error('No valid ASPE URI provided'))
|
||||
return
|
||||
}
|
||||
|
||||
const url = `https://${match[1]}/.well-known/aspe/id/${match[2].toUpperCase()}`
|
||||
|
||||
axios.get(url, {
|
||||
headers: {
|
||||
Accept: 'application/asp+jwt',
|
||||
'User-Agent': `doipjs/${version}`
|
||||
},
|
||||
validateStatus: (status) => status >= 200 && status < 400
|
||||
})
|
||||
.then(async res => await parseProfileJws(res.data, data.aspeUri))
|
||||
.then(profile =>
|
||||
profile.personas.flatMap(p => { return p.claims.map(c => c._uri) })
|
||||
)
|
||||
.then(res => {
|
||||
resolve({
|
||||
claims: res
|
||||
})
|
||||
})
|
||||
.catch(e => {
|
||||
reject(e)
|
||||
})
|
||||
})
|
||||
|
||||
return Promise.race([fetchPromise, timeoutPromise]).then((result) => {
|
||||
clearTimeout(timeoutHandle)
|
||||
return result
|
||||
})
|
||||
}
|
|
@ -14,6 +14,7 @@ See the License for the specific language governing permissions and
|
|||
limitations under the License.
|
||||
*/
|
||||
export * as activitypub from './activitypub.js'
|
||||
export * as aspe from './aspe.js'
|
||||
export * as dns from './dns.js'
|
||||
export * as graphql from './graphql.js'
|
||||
export * as http from './http.js'
|
||||
|
|
|
@ -14,6 +14,7 @@ See the License for the specific language governing permissions and
|
|||
limitations under the License.
|
||||
*/
|
||||
export * as activitypub from './activitypub.js'
|
||||
export * as aspe from './aspe.js'
|
||||
export * as graphql from './graphql.js'
|
||||
export * as http from './http.js'
|
||||
export * as matrix from './matrix.js'
|
||||
|
|
86
src/serviceProviders/aspe.js
Normal file
86
src/serviceProviders/aspe.js
Normal file
|
@ -0,0 +1,86 @@
|
|||
/*
|
||||
Copyright 2024 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.
|
||||
*/
|
||||
import isFQDN from 'validator/lib/isFQDN.js'
|
||||
import * as E from '../enums.js'
|
||||
import { ServiceProvider } from '../serviceProvider.js'
|
||||
|
||||
export const reURI = /^aspe:([a-zA-Z0-9.\-_]*):([a-zA-Z0-9]*)/
|
||||
|
||||
/**
|
||||
* @function
|
||||
* @param {string} uri
|
||||
*/
|
||||
export function processURI (uri) {
|
||||
const match = uri.match(reURI)
|
||||
|
||||
if (!isFQDN(match[1])) {
|
||||
return null
|
||||
}
|
||||
|
||||
return new ServiceProvider({
|
||||
about: {
|
||||
id: 'aspe',
|
||||
name: 'ASPE'
|
||||
},
|
||||
profile: {
|
||||
display: match[1],
|
||||
uri,
|
||||
qr: null
|
||||
},
|
||||
claim: {
|
||||
uriRegularExpression: reURI.toString(),
|
||||
uriIsAmbiguous: false
|
||||
},
|
||||
proof: {
|
||||
request: {
|
||||
uri: null,
|
||||
fetcher: E.Fetcher.ASPE,
|
||||
accessRestriction: E.ProofAccessRestriction.NONE,
|
||||
data: {
|
||||
aspeUri: uri
|
||||
}
|
||||
},
|
||||
response: {
|
||||
format: E.ProofFormat.JSON
|
||||
},
|
||||
target: [{
|
||||
format: E.ClaimFormat.URI,
|
||||
encoding: E.EntityEncodingFormat.PLAIN,
|
||||
relation: E.ClaimRelation.CONTAINS,
|
||||
path: ['claims']
|
||||
}]
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
export const tests = [
|
||||
{
|
||||
uri: 'aspe:domain.tld:abc123def456',
|
||||
shouldMatch: true
|
||||
},
|
||||
{
|
||||
uri: 'aspe:domain.tld',
|
||||
shouldMatch: false
|
||||
},
|
||||
{
|
||||
uri: 'dns:domain.tld',
|
||||
shouldMatch: false
|
||||
},
|
||||
{
|
||||
uri: 'https://domain.tld',
|
||||
shouldMatch: false
|
||||
}
|
||||
]
|
|
@ -13,6 +13,7 @@ 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.
|
||||
*/
|
||||
import * as aspe from './aspe.js'
|
||||
import * as openpgp from './openpgp.js'
|
||||
import * as dns from './dns.js'
|
||||
import * as irc from './irc.js'
|
||||
|
@ -38,6 +39,7 @@ import * as keybase from './keybase.js'
|
|||
import * as opencollective from './opencollective.js'
|
||||
|
||||
const _data = {
|
||||
aspe,
|
||||
openpgp,
|
||||
dns,
|
||||
irc,
|
||||
|
|
Loading…
Reference in a new issue