paricafe/src/remote/activitypub/resolver.ts

84 lines
2 KiB
TypeScript
Raw Normal View History

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>;
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,
proxy: config.proxy.getOrElse(null),
timeout: this.timeout,
2018-04-01 07:56:11 -05:00
headers: {
2018-09-04 03:44:21 -05:00
'User-Agent': config.user_agent,
2018-04-01 07:56:11 -05:00
Accept: 'application/activity+json, application/ld+json'
},
json: true
}).catch(e => {
2019-02-05 09:01:37 -06:00
logger.error(`request error: ${e.message}`);
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
}
}