2018-04-05 04:08:51 -05:00
|
|
|
import * as request from 'request-promise-native';
|
|
|
|
import { IObject } from './type';
|
2018-09-04 03:44:21 -05:00
|
|
|
import config from '../../config';
|
2019-02-02 13:18:27 -06:00
|
|
|
import { apLogger } from './logger';
|
2018-04-04 09:12:35 -05:00
|
|
|
|
2019-02-02 13:18:27 -06:00
|
|
|
export const logger = apLogger.createSubLogger('resolver');
|
2018-03-31 05:55:00 -05:00
|
|
|
|
2018-04-01 07:56:11 -05:00
|
|
|
export default class Resolver {
|
2018-04-04 09:12:35 -05:00
|
|
|
private history: Set<string>;
|
2018-10-04 11:58:41 -05:00
|
|
|
private timeout = 10 * 1000;
|
2018-04-01 07:56:11 -05:00
|
|
|
|
2018-04-04 09:12:35 -05:00
|
|
|
constructor() {
|
|
|
|
this.history = new Set();
|
2018-03-31 05:55:00 -05:00
|
|
|
}
|
|
|
|
|
2018-06-18 00:28:43 -05:00
|
|
|
public async resolveCollection(value: any) {
|
2018-04-04 09:12:35 -05:00
|
|
|
const collection = typeof value === 'string'
|
|
|
|
? await this.resolve(value)
|
|
|
|
: value;
|
|
|
|
|
|
|
|
switch (collection.type) {
|
2019-02-05 09:01:37 -06:00
|
|
|
case 'Collection': {
|
|
|
|
collection.objects = collection.items;
|
|
|
|
break;
|
|
|
|
}
|
2018-04-04 09:12:35 -05:00
|
|
|
|
2019-02-05 09:01:37 -06:00
|
|
|
case 'OrderedCollection': {
|
|
|
|
collection.objects = collection.orderedItems;
|
|
|
|
break;
|
|
|
|
}
|
2018-04-04 09:12:35 -05:00
|
|
|
|
2019-02-05 09:01:37 -06:00
|
|
|
default: {
|
|
|
|
logger.error(`unknown collection type: ${collection.type}`);
|
|
|
|
throw new Error(`unknown collection type: ${collection.type}`);
|
|
|
|
}
|
2018-04-04 09:12:35 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return collection;
|
|
|
|
}
|
|
|
|
|
2018-06-18 00:28:43 -05:00
|
|
|
public async resolve(value: any): Promise<IObject> {
|
2018-04-05 08:49:41 -05:00
|
|
|
if (value == null) {
|
2019-02-05 09:01:37 -06:00
|
|
|
logger.error('resolvee is null (or undefined)');
|
2018-04-05 08:49:41 -05:00
|
|
|
throw new Error('resolvee is null (or undefined)');
|
|
|
|
}
|
|
|
|
|
2018-04-01 07:56:11 -05:00
|
|
|
if (typeof value !== 'string') {
|
2018-04-04 09:12:35 -05:00
|
|
|
return value;
|
2018-04-01 07:56:11 -05:00
|
|
|
}
|
2018-03-31 05:55:00 -05:00
|
|
|
|
2018-04-04 09:12:35 -05:00
|
|
|
if (this.history.has(value)) {
|
2019-02-05 09:01:37 -06:00
|
|
|
logger.error(`cannot resolve already resolved one`);
|
2018-04-04 09:12:35 -05:00
|
|
|
throw new Error('cannot resolve already resolved one');
|
|
|
|
}
|
2018-03-31 05:55:00 -05:00
|
|
|
|
2018-04-04 09:12:35 -05:00
|
|
|
this.history.add(value);
|
2018-03-31 05:55:00 -05:00
|
|
|
|
2018-04-01 07:56:11 -05:00
|
|
|
const object = await request({
|
|
|
|
url: value,
|
2019-02-07 06:02:33 -06:00
|
|
|
proxy: config.proxy,
|
2018-10-04 11:58:41 -05:00
|
|
|
timeout: this.timeout,
|
2018-04-01 07:56:11 -05:00
|
|
|
headers: {
|
2019-02-23 21:53:22 -06:00
|
|
|
'User-Agent': config.userAgent,
|
2018-04-01 07:56:11 -05:00
|
|
|
Accept: 'application/activity+json, application/ld+json'
|
|
|
|
},
|
|
|
|
json: true
|
2018-09-23 17:35:39 -05:00
|
|
|
}).catch(e => {
|
2019-02-05 09:01:37 -06:00
|
|
|
logger.error(`request error: ${e.message}`);
|
2018-09-23 17:35:39 -05:00
|
|
|
throw new Error(`request error: ${e.message}`);
|
2018-04-01 07:56:11 -05:00
|
|
|
});
|
2018-03-31 05:55:00 -05:00
|
|
|
|
2018-04-01 07:56:11 -05:00
|
|
|
if (object === null || (
|
|
|
|
Array.isArray(object['@context']) ?
|
|
|
|
!object['@context'].includes('https://www.w3.org/ns/activitystreams') :
|
|
|
|
object['@context'] !== 'https://www.w3.org/ns/activitystreams'
|
|
|
|
)) {
|
2019-02-02 13:18:27 -06:00
|
|
|
logger.error(`invalid response: ${value}`);
|
2018-04-04 09:12:35 -05:00
|
|
|
throw new Error('invalid response');
|
2018-03-31 05:55:00 -05:00
|
|
|
}
|
|
|
|
|
2018-04-04 09:12:35 -05:00
|
|
|
return object;
|
2018-03-31 05:55:00 -05:00
|
|
|
}
|
|
|
|
}
|