2021-03-25 07:31:29 -06:00
|
|
|
/*
|
|
|
|
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.
|
|
|
|
*/
|
2021-07-09 15:44:52 -06:00
|
|
|
const jsEnv = require('browser-or-node')
|
2021-03-25 07:31:29 -06:00
|
|
|
|
2021-04-22 07:14:21 -06:00
|
|
|
/**
|
|
|
|
* @module fetcher/xmpp
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The request's timeout value in milliseconds
|
|
|
|
* @constant {number} timeout
|
|
|
|
*/
|
2021-04-15 14:30:11 -06:00
|
|
|
module.exports.timeout = 5000
|
|
|
|
|
2021-04-26 04:28:43 -06:00
|
|
|
if (jsEnv.isNode) {
|
|
|
|
const { client, xml } = require('@xmpp/client')
|
|
|
|
const debug = require('@xmpp/debug')
|
|
|
|
const validator = require('validator')
|
2021-07-09 15:44:52 -06:00
|
|
|
|
|
|
|
let xmpp = null
|
|
|
|
let iqCaller = null
|
|
|
|
|
2021-04-26 04:28:43 -06:00
|
|
|
const xmppStart = async (service, username, password) => {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const xmpp = client({
|
|
|
|
service: service,
|
|
|
|
username: username,
|
2021-07-09 15:44:52 -06:00
|
|
|
password: password
|
2021-04-26 04:28:43 -06:00
|
|
|
})
|
|
|
|
if (process.env.NODE_ENV !== 'production') {
|
|
|
|
debug(xmpp, true)
|
|
|
|
}
|
|
|
|
const { iqCaller } = xmpp
|
|
|
|
xmpp.start()
|
2022-12-12 08:20:55 -07:00
|
|
|
xmpp.on('online', _ => {
|
2021-04-26 04:28:43 -06:00
|
|
|
resolve({ xmpp: xmpp, iqCaller: iqCaller })
|
|
|
|
})
|
2022-12-12 08:20:55 -07:00
|
|
|
xmpp.on('error', error => {
|
2021-04-26 04:28:43 -06:00
|
|
|
reject(error)
|
|
|
|
})
|
2021-03-25 07:31:29 -06:00
|
|
|
})
|
2021-04-26 04:28:43 -06:00
|
|
|
}
|
2021-07-09 15:44:52 -06:00
|
|
|
|
2021-04-26 04:28:43 -06:00
|
|
|
/**
|
|
|
|
* Execute a fetch request
|
|
|
|
* @function
|
|
|
|
* @async
|
|
|
|
* @param {object} data - Data used in the request
|
|
|
|
* @param {string} data.id - The identifier of the targeted account
|
|
|
|
* @param {object} opts - Options used to enable the request
|
|
|
|
* @param {string} opts.claims.xmpp.service - The server hostname on which the library can log in
|
|
|
|
* @param {string} opts.claims.xmpp.username - The username used to log in
|
|
|
|
* @param {string} opts.claims.xmpp.password - The password used to log in
|
|
|
|
* @returns {object}
|
|
|
|
*/
|
|
|
|
module.exports.fn = async (data, opts) => {
|
2021-07-09 15:44:52 -06:00
|
|
|
try {
|
|
|
|
validator.isFQDN(opts.claims.xmpp.service)
|
|
|
|
validator.isAscii(opts.claims.xmpp.username)
|
|
|
|
validator.isAscii(opts.claims.xmpp.password)
|
|
|
|
} catch (err) {
|
|
|
|
throw new Error(`XMPP fetcher was not set up properly (${err.message})`)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!xmpp || xmpp.status !== 'online') {
|
|
|
|
const xmppStartRes = await xmppStart(
|
|
|
|
opts.claims.xmpp.service,
|
|
|
|
opts.claims.xmpp.username,
|
|
|
|
opts.claims.xmpp.password
|
|
|
|
)
|
|
|
|
xmpp = xmppStartRes.xmpp
|
|
|
|
iqCaller = xmppStartRes.iqCaller
|
|
|
|
}
|
|
|
|
|
2021-04-26 04:28:43 -06:00
|
|
|
let timeoutHandle
|
|
|
|
const timeoutPromise = new Promise((resolve, reject) => {
|
|
|
|
timeoutHandle = setTimeout(
|
|
|
|
() => reject(new Error('Request was timed out')),
|
|
|
|
data.fetcherTimeout ? data.fetcherTimeout : module.exports.timeout
|
|
|
|
)
|
2021-03-25 07:31:29 -06:00
|
|
|
})
|
2021-07-09 15:44:52 -06:00
|
|
|
|
|
|
|
const fetchPromise = new Promise((resolve, reject) => {
|
2022-12-04 09:28:04 -07:00
|
|
|
(async () => {
|
|
|
|
let completed = false
|
2022-12-12 08:20:55 -07:00
|
|
|
const proofs = []
|
|
|
|
|
|
|
|
// Try the ariadne-id pubsub request
|
|
|
|
if (!completed) {
|
|
|
|
try {
|
|
|
|
const response = await iqCaller.request(
|
|
|
|
xml('iq', { type: 'get', to: data.id }, xml('pubsub', 'http://jabber.org/protocol/pubsub', xml('items', { node: 'http://ariadne.id/protocol/proof' }))),
|
|
|
|
30 * 1000
|
|
|
|
)
|
|
|
|
|
|
|
|
// Traverse the XML response
|
|
|
|
response.getChild('pubsub').getChildren('items').forEach(items => {
|
|
|
|
if (items.attrs.node === 'http://ariadne.id/protocol/proof') {
|
|
|
|
items.getChildren('item').forEach(item => {
|
|
|
|
proofs.push(item.getChildText('value'))
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
resolve(proofs)
|
|
|
|
completed = true
|
|
|
|
} catch (_) {}
|
2021-04-26 04:28:43 -06:00
|
|
|
}
|
2022-12-04 09:28:04 -07:00
|
|
|
|
2022-12-12 08:20:55 -07:00
|
|
|
// Try the vcard4 pubsub request [backward compatibility]
|
2022-12-04 09:28:04 -07:00
|
|
|
if (!completed) {
|
|
|
|
try {
|
|
|
|
const response = await iqCaller.request(
|
|
|
|
xml('iq', { type: 'get', to: data.id }, xml('pubsub', 'http://jabber.org/protocol/pubsub', xml('items', { node: 'urn:xmpp:vcard4', max_items: '1' }))),
|
|
|
|
30 * 1000
|
|
|
|
)
|
|
|
|
|
|
|
|
// Traverse the XML response
|
|
|
|
response.getChild('pubsub').getChildren('items').forEach(items => {
|
|
|
|
if (items.attrs.node === 'urn:xmpp:vcard4') {
|
|
|
|
items.getChildren('item').forEach(item => {
|
|
|
|
if (item.attrs.id === 'current') {
|
|
|
|
const itemVcard = item.getChild('vcard', 'urn:ietf:params:xml:ns:vcard-4.0')
|
|
|
|
// Find the vCard URLs
|
|
|
|
itemVcard.getChildren('url').forEach(url => {
|
2022-12-12 08:20:55 -07:00
|
|
|
proofs.push(url.getChildText('uri'))
|
2022-12-04 09:28:04 -07:00
|
|
|
})
|
|
|
|
// Find the vCard notes
|
|
|
|
itemVcard.getChildren('note').forEach(note => {
|
2022-12-12 08:20:55 -07:00
|
|
|
proofs.push(note.getChildText('text'))
|
2022-12-04 09:28:04 -07:00
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2022-12-12 08:20:55 -07:00
|
|
|
resolve(proofs)
|
2022-12-04 09:28:04 -07:00
|
|
|
completed = true
|
|
|
|
} catch (_) {}
|
|
|
|
}
|
|
|
|
|
2022-12-12 08:20:55 -07:00
|
|
|
// Try the vcard-temp IQ request [backward compatibility]
|
2022-12-04 09:28:04 -07:00
|
|
|
if (!completed) {
|
|
|
|
try {
|
|
|
|
const response = await iqCaller.request(
|
|
|
|
xml('iq', { type: 'get', to: data.id }, xml('vCard', 'vcard-temp')),
|
|
|
|
30 * 1000
|
|
|
|
)
|
|
|
|
|
|
|
|
// Find the vCard URLs
|
|
|
|
response.getChild('vCard', 'vcard-temp').getChildren('URL').forEach(url => {
|
2022-12-12 08:20:55 -07:00
|
|
|
proofs.push(url.children[0])
|
2022-12-04 09:28:04 -07:00
|
|
|
})
|
|
|
|
// Find the vCard notes
|
|
|
|
response.getChild('vCard', 'vcard-temp').getChildren('NOTE').forEach(note => {
|
2022-12-12 08:20:55 -07:00
|
|
|
proofs.push(note.children[0])
|
2022-12-04 09:28:04 -07:00
|
|
|
})
|
|
|
|
response.getChild('vCard', 'vcard-temp').getChildren('DESC').forEach(note => {
|
2022-12-12 08:20:55 -07:00
|
|
|
proofs.push(note.children[0])
|
2022-12-04 09:28:04 -07:00
|
|
|
})
|
|
|
|
|
2022-12-12 08:20:55 -07:00
|
|
|
resolve(proofs)
|
2022-12-04 09:28:04 -07:00
|
|
|
completed = true
|
|
|
|
} catch (error) {
|
|
|
|
reject(error)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-26 04:28:43 -06:00
|
|
|
xmpp.stop()
|
2022-12-04 09:28:04 -07:00
|
|
|
})()
|
2021-04-26 04:28:43 -06:00
|
|
|
})
|
2021-07-09 15:44:52 -06:00
|
|
|
|
2021-04-26 04:28:43 -06:00
|
|
|
return Promise.race([fetchPromise, timeoutPromise]).then((result) => {
|
|
|
|
clearTimeout(timeoutHandle)
|
|
|
|
return result
|
|
|
|
})
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
module.exports.fn = null
|
2021-03-25 08:37:30 -06:00
|
|
|
}
|