const WebHandlerModule = require('../../../web_handler_module'); const Config = require('../../../config').get; const { Errors, ErrorReasons } = require('../../../enig_error'); const { WellKnownLocations } = require('../web'); const { getUserProfileTemplatedBody, DefaultProfileTemplate, } = require('../../../activitypub/util'); const Endpoints = require('../../../activitypub/endpoint'); const EngiAssert = require('../../../enigma_assert'); const User = require('../../../user'); const UserProps = require('../../../user_property'); const ActivityPubSettings = require('../../../activitypub/settings'); const { getFullUrl, buildUrl, getWebDomain } = require('../../../web_util'); // deps const _ = require('lodash'); const Actor = require('../../../activitypub/actor'); exports.moduleInfo = { name: 'WebFinger', desc: 'A simple WebFinger Handler.', author: 'NuSkooler, CognitiveGears', packageName: 'codes.l33t.enigma.web.handler.webfinger', }; // // WebFinger: https://www.rfc-editor.org/rfc/rfc7033 // exports.getModule = class WebFingerWebHandler extends WebHandlerModule { constructor() { super(); } init(webServer, cb) { // we rely on the web server this.webServer = webServer; EngiAssert(webServer, 'WebFinger Web Handler init without webServer'); this.log = webServer.logger().child({ webHandler: 'WebFinger' }); const domain = getWebDomain(); if (!domain) { return cb(Errors.UnexpectedState('Web server does not have "domain" set')); } this.acceptedResourceRegExps = [ // acct:NAME@our.domain.tld // https://www.rfc-editor.org/rfc/rfc7565 new RegExp(`^acct:(.+)@${domain}$`), // profile page // https://webfinger.net/rel/profile-page/ new RegExp(`^${buildUrl(WellKnownLocations.Internal + '/wf/@')}(.+)$`), // self URL new RegExp(`^${buildUrl(WellKnownLocations.Internal + '/ap/users/')}(.+)$`), ]; this.webServer.addRoute({ method: 'GET', // https://www.rfc-editor.org/rfc/rfc7033.html#section-10.1 path: /^\/\.well-known\/webfinger\/?\?/, handler: this._webFingerRequestHandler.bind(this), }); this.webServer.addRoute({ method: 'GET', path: /^\/_enig\/wf\/@.+$/, handler: this._profileRequestHandler.bind(this), }); return cb(null); } _profileRequestHandler(req, resp) { // Profile requests do not have an Actor ID available const profileQuery = getFullUrl(req).toString(); const accountName = this._getAccountName(profileQuery); if (!accountName) { this.log.warn( `Failed to parse "account name" for profile query: ${profileQuery}` ); return this.webServer.resourceNotFound(resp); } this._localUserFromWebFingerAccountName(accountName, (err, localUser) => { if (err) { this.log.warn( { error: err.message, type: 'Profile', accountName }, 'Could not fetch profile for WebFinger request' ); return this.webServer.resourceNotFound(resp); } let templateFile = _.get( Config(), 'contentServers.web.handlers.webFinger.profileTemplate' ); if (templateFile) { templateFile = this.webServer.resolveTemplatePath(templateFile); } Actor.fromLocalUser(localUser, this.webServer, (err, localActor) => { if (err) { return this.webServer.internalServerError(resp, err); } getUserProfileTemplatedBody( this.webServer, templateFile, localUser, localActor, DefaultProfileTemplate, 'text/plain', (err, body, contentType) => { if (err) { return this.webServer.resourceNotFound(resp); } const headers = { 'Content-Type': contentType, 'Content-Length': Buffer(body).length, }; resp.writeHead(200, headers); return resp.end(body); } ); }); }); } _webFingerRequestHandler(req, resp) { const url = getFullUrl(req); const resource = url.searchParams.get('resource'); if (!resource) { return this.webServer.respondWithError( resp, 400, '"resource" is required', 'Missing "resource"' ); } const accountName = this._getAccountName(resource); if (!accountName || accountName.length < 1) { this.log.warn(`Failed to parse "account name" for resource: ${resource}`); return this.webServer.resourceNotFound(resp); } this._localUserFromWebFingerAccountName(accountName, (err, localUser) => { if (err) { this.log.warn( { url: req.url, error: err.message, type: 'WebFinger' }, `No account for "${accountName}" could be retrieved` ); return this.webServer.resourceNotFound(resp); } const domain = getWebDomain(); const body = JSON.stringify({ subject: `acct:${localUser.username}@${domain}`, aliases: [Endpoints.profile(localUser), Endpoints.actorId(localUser)], links: [ this._profilePageLink(localUser), this._selfLink(localUser), this._subscribeLink(), ], }); const headers = { 'Content-Type': 'application/jrd+json', 'Content-Length': Buffer(body).length, }; resp.writeHead(200, headers); return resp.end(body); }); } _localUserFromWebFingerAccountName(accountName, cb) { if (accountName.startsWith('@')) { accountName = accountName.slice(1); } User.getUserIdAndName(accountName, (err, userId) => { if (err) { return cb(err); } User.getUser(userId, (err, user) => { if (err) { return cb(err); } const accountStatus = user.getPropertyAsNumber(UserProps.AccountStatus); if ( User.AccountStatus.disabled == accountStatus || User.AccountStatus.inactive == accountStatus ) { return cb( Errors.AccessDenied('Account disabled', ErrorReasons.Disabled) ); } const activityPubSettings = ActivityPubSettings.fromUser(user); if (!activityPubSettings.enabled) { return cb(Errors.AccessDenied('ActivityPub is not enabled for user')); } return cb(null, user); }); }); } _profilePageLink(user) { const href = Endpoints.profile(user); return { rel: 'http://webfinger.net/rel/profile-page', type: 'text/plain', href, }; } _userActorId(user) { return Endpoints.actorId(user); } // :TODO: only if ActivityPub is enabled _selfLink(user) { const href = Endpoints.actorId(user); return { rel: 'self', type: 'application/activity+json', href, }; } // :TODO: only if ActivityPub is enabled _subscribeLink() { return { rel: 'http://ostatus.org/schema/1.0/subscribe', template: buildUrl( WellKnownLocations.Internal + '/ap/authorize_interaction?uri={uri}' ), }; } _getAccountName(resource) { for (const re of this.acceptedResourceRegExps) { const m = resource.match(re); if (m && m.length === 2) { return m[1]; } } } };