summaryrefslogtreecommitdiff
path: root/src/server/webfinger.ts
blob: dbf0999f3e3afcfe1978ace0de4361e7bd6e2909 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
import * as express from 'express';

import config from '../config';
import parseAcct from '../acct/parse';
import User from '../models/user';

const app = express.Router();

app.get('/.well-known/webfinger', async (req, res) => {
	if (typeof req.query.resource !== 'string') {
		return res.sendStatus(400);
	}

	const resourceLower = req.query.resource.toLowerCase();
	const webPrefix = config.url.toLowerCase() + '/@';
	let acctLower;

	if (resourceLower.startsWith(webPrefix)) {
		acctLower = resourceLower.slice(webPrefix.length);
	} else if (resourceLower.startsWith('acct:')) {
		acctLower = resourceLower.slice('acct:'.length);
	} else {
		acctLower = resourceLower;
	}

	const parsedAcctLower = parseAcct(acctLower);
	if (![null, config.host.toLowerCase()].includes(parsedAcctLower.host)) {
		return res.sendStatus(422);
	}

	const user = await User.findOne({ usernameLower: parsedAcctLower.username, host: null });
	if (user === null) {
		return res.sendStatus(404);
	}

	return res.json({
		subject: `acct:${user.username}@${config.host}`,
		links: [{
			rel: 'self',
			type: 'application/activity+json',
			href: `${config.url}/users/${user._id}`
		}, {
			rel: 'http://webfinger.net/rel/profile-page',
			type: 'text/html',
			href: `${config.url}/@${user.username}`
		}, {
			rel: 'http://ostatus.org/schema/1.0/subscribe',
			template: `${config.url}/authorize-follow?acct={uri}`
		}]
	});
});

export default app;