From 05ff9ae93f76176091f18ba6c22378e81d633249 Mon Sep 17 00:00:00 2001 From: Hazelnoot Date: Wed, 18 Dec 2024 10:29:25 -0500 Subject: [PATCH] check for cross-domain redirects that bounce from an allowed domain to a blocked domain --- .../src/core/activitypub/ApResolverService.ts | 23 +++++++++++++------ 1 file changed, 16 insertions(+), 7 deletions(-) diff --git a/packages/backend/src/core/activitypub/ApResolverService.ts b/packages/backend/src/core/activitypub/ApResolverService.ts index e56b9ebc99..c82a9be3b1 100644 --- a/packages/backend/src/core/activitypub/ApResolverService.ts +++ b/packages/backend/src/core/activitypub/ApResolverService.ts @@ -104,7 +104,7 @@ export class Resolver { } if (!this.utilityService.isFederationAllowedHost(host)) { - throw new UnrecoverableError(`instance is blocked: ${value}`); + throw new UnrecoverableError(`cannot fetch AP object ${value}: blocked instance ${host}`); } if (this.config.signToActivityPubGet && !this.user) { @@ -123,16 +123,25 @@ export class Resolver { throw new UnrecoverableError(`invalid AP object ${value}: does not have ActivityStreams context`); } - // HttpRequestService / ApRequestService have already checked that - // `object.id` or `object.url` matches the URL used to fetch the - // object after redirects; here we double-check that no redirects - // bounced between hosts + // Since redirects are allowed, we cannot safely validate an anonymous object. + // Reject any responses without an ID, as all other checks depend on that value. if (object.id == null) { throw new UnrecoverableError(`invalid AP object ${value}: missing id`); } - if (this.utilityService.punyHostPSLDomain(object.id) !== this.utilityService.punyHostPSLDomain(value)) { - throw new UnrecoverableError(`invalid AP object ${value}: id ${object.id} has different host`); + // We allow some limited cross-domain redirects, which means the host may have changed during fetch. + // Additional checks are needed to validate the scope of cross-domain redirects. + const finalHost = this.utilityService.extractDbHost(object.id); + if (finalHost !== host) { + // Make sure the redirect stayed within the same authority. + if (this.utilityService.punyHostPSLDomain(object.id) !== this.utilityService.punyHostPSLDomain(value)) { + throw new UnrecoverableError(`invalid AP object ${value}: id ${object.id} has different host`); + } + + // Check if the redirect bounce from [allowed domain] to [blocked domain]. + if (!this.utilityService.isFederationAllowedHost(finalHost)) { + throw new UnrecoverableError(`cannot fetch AP object ${value}: redirected to blocked instance ${finalHost}`); + } } return object;