forked from Mirrors/doipjs
193 lines
4.9 KiB
JavaScript
193 lines
4.9 KiB
JavaScript
/*
|
|
Copyright 2021 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.
|
|
*/
|
|
const jsEnv = require('browser-or-node')
|
|
const fetcher = require('./fetcher')
|
|
const utils = require('./utils')
|
|
const E = require('./enums')
|
|
|
|
/**
|
|
* @module proofs
|
|
*/
|
|
|
|
/**
|
|
* Delegate the proof request to the correct fetcher.
|
|
* This method uses the current environment (browser/node), certain values from
|
|
* the `data` parameter and the proxy policy set in the `opts` parameter to
|
|
* choose the right approach to fetch the proof. An error will be thrown if no
|
|
* approach is possible.
|
|
* @async
|
|
* @param {object} data - Data from a claim definition
|
|
* @param {object} opts - Options to enable the request
|
|
* @returns {Promise<object|string>}
|
|
*/
|
|
const fetch = (data, opts) => {
|
|
switch (data.proof.request.fetcher) {
|
|
case E.Fetcher.HTTP:
|
|
data.proof.request.data.format = data.proof.request.format
|
|
break
|
|
|
|
default:
|
|
break
|
|
}
|
|
|
|
if (jsEnv.isNode) {
|
|
return handleNodeRequests(data, opts)
|
|
}
|
|
|
|
return handleBrowserRequests(data, opts)
|
|
}
|
|
|
|
const handleBrowserRequests = (data, opts) => {
|
|
switch (opts.proxy.policy) {
|
|
case E.ProxyPolicy.ALWAYS:
|
|
return createProxyRequestPromise(data, opts)
|
|
break
|
|
|
|
case E.ProxyPolicy.NEVER:
|
|
switch (data.proof.request.access) {
|
|
case E.ProofAccess.GENERIC:
|
|
case E.ProofAccess.GRANTED:
|
|
return createDefaultRequestPromise(data, opts)
|
|
break
|
|
case E.ProofAccess.NOCORS:
|
|
case E.ProofAccess.SERVER:
|
|
throw new Error(
|
|
'Impossible to fetch proof (bad combination of service access and proxy policy)'
|
|
)
|
|
break
|
|
default:
|
|
throw new Error('Invalid proof access value')
|
|
break
|
|
}
|
|
break
|
|
|
|
case E.ProxyPolicy.ADAPTIVE:
|
|
switch (data.proof.request.access) {
|
|
case E.ProofAccess.GENERIC:
|
|
return createDefaultRequestPromise(data, opts)
|
|
break
|
|
case E.ProofAccess.NOCORS:
|
|
return createProxyRequestPromise(data, opts)
|
|
break
|
|
case E.ProofAccess.GRANTED:
|
|
return createFallbackRequestPromise(data, opts)
|
|
break
|
|
case E.ProofAccess.SERVER:
|
|
return createProxyRequestPromise(data, opts)
|
|
break
|
|
default:
|
|
throw new Error('Invalid proof access value')
|
|
break
|
|
}
|
|
break
|
|
|
|
default:
|
|
throw new Error('Invalid proxy policy')
|
|
break
|
|
}
|
|
}
|
|
|
|
const handleNodeRequests = (data, opts) => {
|
|
switch (opts.proxy.policy) {
|
|
case E.ProxyPolicy.ALWAYS:
|
|
return createProxyRequestPromise(data, opts)
|
|
break
|
|
|
|
case E.ProxyPolicy.NEVER:
|
|
return createDefaultRequestPromise(data, opts)
|
|
break
|
|
|
|
case E.ProxyPolicy.ADAPTIVE:
|
|
return createFallbackRequestPromise(data, opts)
|
|
break
|
|
|
|
default:
|
|
throw new Error('Invalid proxy policy')
|
|
break
|
|
}
|
|
}
|
|
|
|
const createDefaultRequestPromise = (data, opts) => {
|
|
return new Promise((resolve, reject) => {
|
|
fetcher[data.proof.request.fetcher]
|
|
.fn(data.proof.request.data, opts)
|
|
.then((res) => {
|
|
return resolve({
|
|
fetcher: data.proof.request.fetcher,
|
|
data: data,
|
|
viaProxy: false,
|
|
result: res,
|
|
})
|
|
})
|
|
.catch((err) => {
|
|
return reject(err)
|
|
})
|
|
})
|
|
}
|
|
|
|
const createProxyRequestPromise = (data, opts) => {
|
|
return new Promise((resolve, reject) => {
|
|
let proxyUrl
|
|
try {
|
|
proxyUrl = utils.generateProxyURL(
|
|
data.proof.request.fetcher,
|
|
data.proof.request.data,
|
|
opts
|
|
)
|
|
} catch (err) {
|
|
reject(err)
|
|
}
|
|
|
|
const requestData = {
|
|
url: proxyUrl,
|
|
format: data.proof.request.format,
|
|
fetcherTimeout: fetcher[data.proof.request.fetcher].timeout,
|
|
}
|
|
fetcher.http
|
|
.fn(requestData, opts)
|
|
.then((res) => {
|
|
return resolve({
|
|
fetcher: 'http',
|
|
data: data,
|
|
viaProxy: true,
|
|
result: res,
|
|
})
|
|
})
|
|
.catch((err) => {
|
|
return reject(err)
|
|
})
|
|
})
|
|
}
|
|
|
|
const createFallbackRequestPromise = (data, opts) => {
|
|
return new Promise((resolve, reject) => {
|
|
createDefaultRequestPromise(data, opts)
|
|
.then((res) => {
|
|
return resolve(res)
|
|
})
|
|
.catch((err1) => {
|
|
createProxyRequestPromise(data, opts)
|
|
.then((res) => {
|
|
return resolve(res)
|
|
})
|
|
.catch((err2) => {
|
|
return reject([err1, err2])
|
|
})
|
|
})
|
|
})
|
|
}
|
|
|
|
exports.fetch = fetch
|