Loading CHANGES.md +5 −0 Original line number Diff line number Diff line Loading @@ -58,9 +58,13 @@ To be released. - Changed the `href` field optional from the `Link` interface according to [RFC 7033 Section 4.4.4.3]. - Added `Federatable.setWebFingerLinksDispatcher()` method to set additional links to WebFinger. [[#119], [#407] by HyeonseoKim] [FEP-5711]: https://w3id.org/fep/5711 [OStatus 1.0 Draft 2]: https://www.w3.org/community/ostatus/wiki/images/9/93/OStatus_1.0_Draft_2.pdf [RFC 7033 Section 4.4.4.3]: https://datatracker.ietf.org/doc/html/rfc7033#section-4.4.4.3 [#119]: https://github.com/fedify-dev/fedify/issues/119 [#353]: https://github.com/fedify-dev/fedify/issues/353 [#365]: https://github.com/fedify-dev/fedify/pull/365 [#373]: https://github.com/fedify-dev/fedify/issues/373 Loading @@ -68,6 +72,7 @@ To be released. [#381]: https://github.com/fedify-dev/fedify/pull/381 [#402]: https://github.com/fedify-dev/fedify/issues/402 [#404]: https://github.com/fedify-dev/fedify/pull/404 [#407]: https://github.com/fedify-dev/fedify/pull/407 ### @fedify/cli Loading docs/.vitepress/config.mts +1 −0 Original line number Diff line number Diff line Loading @@ -68,6 +68,7 @@ const MANUAL = { { text: "Collections", link: "/manual/collections.md" }, { text: "Object dispatcher", link: "/manual/object.md" }, { text: "Access control", link: "/manual/access-control.md" }, { text: "WebFinger", link: "/manual/webfinger.md" }, { text: "NodeInfo", link: "/manual/nodeinfo.md" }, { text: "Pragmatics", link: "/manual/pragmatics.md" }, { text: "Key–value store", link: "/manual/kv.md" }, Loading docs/manual/actor.md +6 −0 Original line number Diff line number Diff line Loading @@ -478,6 +478,12 @@ WebFinger links Some properties of an `Actor` returned by the actor dispatcher affect responses to WebFinger requests. > [!TIP] > *Since Fedify 1.9.0*, you can also customize WebFinger responses using > `~Federatable.setWebFingerLinksDispatcher()` to add custom links with > the `template` field. See the [WebFinger documentation](./webfinger.md#customizing-webfinger-endpoint) > for details. ### `preferredUsername` *This API is available since Fedify 0.15.0.* Loading docs/manual/webfinger.md 0 → 100644 +250 −0 Original line number Diff line number Diff line --- description: >- WebFinger is a protocol that allows for the discovery of information about people and other entities on the Internet using just their identifier. This section explains how to implement WebFinger endpoints and use the WebFinger client in Fedify. --- WebFinger ========= According to the [WebFinger website]: > WebFinger is used to discover information about people or other entities > on the Internet that are identified by a URI using standard > Hypertext Transfer Protocol (HTTP) methods over a secure transport. > A WebFinger resource returns a JavaScript Object Notation (JSON) object > describing the entity that is queried. The JSON object is referred to as > the JSON Resource Descriptor (JRD). WebFinger is essential for ActivityPub federation. It lets servers discover actor profiles using familiar identifiers like `@user@example.com`. Most ActivityPub implementations, including Mastodon and Misskey, depend on WebFinger for account discovery. > [!NOTE] > Fedify implements WebFinger according to [RFC 7033] specification. [WebFinger website]: https://webfinger.net/ [RFC 7033]: https://datatracker.ietf.org/doc/html/rfc7033 WebFinger schema ---------------- The WebFinger response follows the JSON Resource Descriptor (JRD) format as defined in [RFC 7033]. The main interfaces are: ### `ResourceDescriptor` The main WebFinger response object: `subject` : A URI that identifies the entity that this descriptor describes. This is typically set automatically by Fedify. `aliases` : URIs that identify the same entity as the `subject`. `properties` : Additional key-value properties about the `subject`. `links` : An array of [`Link`] objects pointing to related resources. ### `Link` Represents a link to a related resource: `rel` : *Required.* The link's relation type, which is either a URI or a registered relation type (see [RFC 5988]). `type` : The media type of the target resource (see [RFC 6838]). `href` : A URI pointing to the target resource. `titles` : Human-readable titles describing the link relation. If the language is unknown or unspecified, the key is `"und"`. `properties` : Additional key-value properties about the link relation. `template` : *Since Fedify 1.9.0.* A URI Template ([RFC 6570]) with placeholders for variable substitution. Commonly used for remote follow endpoints where `{uri}` is replaced with the account to follow. ### Common link relations Fedify automatically generates these link relations from actor properties: `"self"` : The actor's ActivityPub profile URI. Uses `application/activity+json` as the media type. `"http://webfinger.net/rel/profile-page"` : The actor's profile page from the `url` property. Uses `text/html` as the media type. `"http://webfinger.net/rel/avatar"` : The actor's avatar from the `icon` property. Additional custom links can be added via WebFinger links dispatcher: `"http://ostatus.org/schema/1.0/subscribe"` : Remote follow endpoint (common in Mastodon). Uses a URI template with `{uri}` placeholder for the account being followed. Example WebFinger response (including both automatic and custom links): ~~~~ json { "subject": "acct:alice@your-domain.com", "links": [ { "rel": "self", "type": "application/activity+json", "href": "https://your-domain.com/users/alice" }, { "rel": "http://webfinger.net/rel/profile-page", "type": "text/html", "href": "https://your-domain.com/@alice" }, { "rel": "http://ostatus.org/schema/1.0/subscribe", "template": "https://your-domain.com/authorize_interaction?uri={uri}" } ] } ~~~~ [`Link`]: https://jsr.io/@fedify/fedify/doc/webfinger/~/Link [RFC 5988]: https://datatracker.ietf.org/doc/html/rfc5988 [RFC 6838]: https://datatracker.ietf.org/doc/html/rfc6838 [RFC 6570]: https://datatracker.ietf.org/doc/html/rfc6570 Customizing WebFinger endpoint ------------------------------ *This API is available since Fedify 1.9.0.* While Fedify automatically handles WebFinger responses for actors registered via `~Federatable.setActorDispatcher()`, you can add custom links using `~Federatable.setWebFingerLinksDispatcher()`. This is useful for adding non-standard links like Mastodon's remote follow endpoint: ~~~~ typescript twoslash // @noErrors: 2345 import { createFederation } from "@fedify/fedify"; const federation = createFederation({ // Omitted for brevity; see the related section for details. }); federation.setWebFingerLinksDispatcher(async (ctx, resource) => { return [ { rel: "http://ostatus.org/schema/1.0/subscribe", template: `https://your-domain.com/@${resource.pathname}/authorize_interaction?uri={uri}` } ]; } ); ~~~~ This gives results like below: ~~~~json { "subject": "acct:alice@your-domain.com", "links": [ { "rel": "self", "type": "application/activity+json", "href": "https://your-domain.com/users/alice" }, { "rel": "http://webfinger.net/rel/profile-page", "type": "text/html", "href": "https://your-domain.com/@alice" }, { "rel": "http://ostatus.org/schema/1.0/subscribe", "template": "https://your-domain.com/@alice@your-domain.com/authorize_interaction?uri={uri}" } ] } ~~~~ The WebFinger links dispatcher receives two parameters: `ctx` : The federation context `resource` : The URL queried via WebFinger > [!TIP] > The WebFinger endpoint is automatically exposed at `/.well-known/webfinger` > by the `Federation.fetch()` method. You don't need to manually handle this > route. > [!NOTE] > Before the introduction of `~Federatable.setWebFingerLinksDispatcher()` in > Fedify 1.9.0, WebFinger responses could only be customized through > `~Federatable.setActorDispatcher()` by setting the actor's `url` property. > This method still works and is sufficient for many use cases, though it > doesn't support the `template` field needed for features like Mastodon's > remote follow functionality. > > See the [WebFinger links section](./actor.md#webfinger-links) in the Actor > documentation for details on customizing WebFinger through actor properties. Looking up WebFinger -------------------- *This API is available since Fedify 1.6.0.* Use `~Context.lookupWebFinger()` to query remote WebFinger endpoints: ~~~~ typescript twoslash import { type Context } from "@fedify/fedify"; const ctx = null as unknown as Context<void>; // ---cut-before--- const webfingerData = await ctx.lookupWebFinger("acct:fedify@hollo.social"); ~~~~ If the lookup fails or the account doesn't exist, the method returns `null`. The returned WebFinger document contains links to various resources associated with the account, such as profile pages, ActivityPub actor URIs, and more: ~~~~ typescript twoslash import { type Context } from "@fedify/fedify"; const ctx = null as unknown as Context<void>; // ---cut-before--- const webfingerData = await ctx.lookupWebFinger("acct:fedify@hollo.social"); // Find the ActivityPub actor URI const activityPubActorLink = webfingerData?.links?.find(link => link.rel === "self" && link.type === "application/activity+json" ); if (activityPubActorLink?.href) { const actor = await ctx.lookupObject(activityPubActorLink.href); // Work with the actor... } ~~~~ > [!NOTE] > In most cases, you can use the higher-level `~Context.lookupObject()` method > which automatically performs WebFinger lookups when given a handle. > Use `~Context.lookupWebFinger()` when you need the raw WebFinger data or > want more direct control over the lookup process. packages/fedify/src/federation/builder.ts +9 −0 Original line number Diff line number Diff line Loading @@ -22,6 +22,7 @@ import type { ObjectAuthorizePredicate, ObjectDispatcher, SharedInboxKeyDispatcher, WebFingerLinksDispatcher, } from "./callback.ts"; import type { Context, RequestContext } from "./context.ts"; import type { Loading @@ -48,6 +49,7 @@ export class FederationBuilderImpl<TContextData> router: Router; actorCallbacks?: ActorCallbacks<TContextData>; nodeInfoDispatcher?: NodeInfoDispatcher<TContextData>; webFingerLinksDispatcher?: WebFingerLinksDispatcher<TContextData>; objectCallbacks: Record<string, ObjectCallbacks<TContextData, string>>; objectTypeIds: Record< string, Loading Loading @@ -148,6 +150,7 @@ export class FederationBuilderImpl<TContextData> ? undefined : { ...this.actorCallbacks }; f.nodeInfoDispatcher = this.nodeInfoDispatcher; f.webFingerLinksDispatcher = this.webFingerLinksDispatcher; f.objectCallbacks = { ...this.objectCallbacks }; f.objectTypeIds = { ...this.objectTypeIds }; f.inboxPath = this.inboxPath; Loading Loading @@ -491,6 +494,12 @@ export class FederationBuilderImpl<TContextData> this.nodeInfoDispatcher = dispatcher; } setWebFingerLinksDispatcher( dispatcher: WebFingerLinksDispatcher<TContextData>, ): void { this.webFingerLinksDispatcher = dispatcher; } setObjectDispatcher<TObject extends Object, TParam extends string>( // deno-lint-ignore no-explicit-any cls: (new (...args: any[]) => TObject) & { typeId: URL }, Loading Loading
CHANGES.md +5 −0 Original line number Diff line number Diff line Loading @@ -58,9 +58,13 @@ To be released. - Changed the `href` field optional from the `Link` interface according to [RFC 7033 Section 4.4.4.3]. - Added `Federatable.setWebFingerLinksDispatcher()` method to set additional links to WebFinger. [[#119], [#407] by HyeonseoKim] [FEP-5711]: https://w3id.org/fep/5711 [OStatus 1.0 Draft 2]: https://www.w3.org/community/ostatus/wiki/images/9/93/OStatus_1.0_Draft_2.pdf [RFC 7033 Section 4.4.4.3]: https://datatracker.ietf.org/doc/html/rfc7033#section-4.4.4.3 [#119]: https://github.com/fedify-dev/fedify/issues/119 [#353]: https://github.com/fedify-dev/fedify/issues/353 [#365]: https://github.com/fedify-dev/fedify/pull/365 [#373]: https://github.com/fedify-dev/fedify/issues/373 Loading @@ -68,6 +72,7 @@ To be released. [#381]: https://github.com/fedify-dev/fedify/pull/381 [#402]: https://github.com/fedify-dev/fedify/issues/402 [#404]: https://github.com/fedify-dev/fedify/pull/404 [#407]: https://github.com/fedify-dev/fedify/pull/407 ### @fedify/cli Loading
docs/.vitepress/config.mts +1 −0 Original line number Diff line number Diff line Loading @@ -68,6 +68,7 @@ const MANUAL = { { text: "Collections", link: "/manual/collections.md" }, { text: "Object dispatcher", link: "/manual/object.md" }, { text: "Access control", link: "/manual/access-control.md" }, { text: "WebFinger", link: "/manual/webfinger.md" }, { text: "NodeInfo", link: "/manual/nodeinfo.md" }, { text: "Pragmatics", link: "/manual/pragmatics.md" }, { text: "Key–value store", link: "/manual/kv.md" }, Loading
docs/manual/actor.md +6 −0 Original line number Diff line number Diff line Loading @@ -478,6 +478,12 @@ WebFinger links Some properties of an `Actor` returned by the actor dispatcher affect responses to WebFinger requests. > [!TIP] > *Since Fedify 1.9.0*, you can also customize WebFinger responses using > `~Federatable.setWebFingerLinksDispatcher()` to add custom links with > the `template` field. See the [WebFinger documentation](./webfinger.md#customizing-webfinger-endpoint) > for details. ### `preferredUsername` *This API is available since Fedify 0.15.0.* Loading
docs/manual/webfinger.md 0 → 100644 +250 −0 Original line number Diff line number Diff line --- description: >- WebFinger is a protocol that allows for the discovery of information about people and other entities on the Internet using just their identifier. This section explains how to implement WebFinger endpoints and use the WebFinger client in Fedify. --- WebFinger ========= According to the [WebFinger website]: > WebFinger is used to discover information about people or other entities > on the Internet that are identified by a URI using standard > Hypertext Transfer Protocol (HTTP) methods over a secure transport. > A WebFinger resource returns a JavaScript Object Notation (JSON) object > describing the entity that is queried. The JSON object is referred to as > the JSON Resource Descriptor (JRD). WebFinger is essential for ActivityPub federation. It lets servers discover actor profiles using familiar identifiers like `@user@example.com`. Most ActivityPub implementations, including Mastodon and Misskey, depend on WebFinger for account discovery. > [!NOTE] > Fedify implements WebFinger according to [RFC 7033] specification. [WebFinger website]: https://webfinger.net/ [RFC 7033]: https://datatracker.ietf.org/doc/html/rfc7033 WebFinger schema ---------------- The WebFinger response follows the JSON Resource Descriptor (JRD) format as defined in [RFC 7033]. The main interfaces are: ### `ResourceDescriptor` The main WebFinger response object: `subject` : A URI that identifies the entity that this descriptor describes. This is typically set automatically by Fedify. `aliases` : URIs that identify the same entity as the `subject`. `properties` : Additional key-value properties about the `subject`. `links` : An array of [`Link`] objects pointing to related resources. ### `Link` Represents a link to a related resource: `rel` : *Required.* The link's relation type, which is either a URI or a registered relation type (see [RFC 5988]). `type` : The media type of the target resource (see [RFC 6838]). `href` : A URI pointing to the target resource. `titles` : Human-readable titles describing the link relation. If the language is unknown or unspecified, the key is `"und"`. `properties` : Additional key-value properties about the link relation. `template` : *Since Fedify 1.9.0.* A URI Template ([RFC 6570]) with placeholders for variable substitution. Commonly used for remote follow endpoints where `{uri}` is replaced with the account to follow. ### Common link relations Fedify automatically generates these link relations from actor properties: `"self"` : The actor's ActivityPub profile URI. Uses `application/activity+json` as the media type. `"http://webfinger.net/rel/profile-page"` : The actor's profile page from the `url` property. Uses `text/html` as the media type. `"http://webfinger.net/rel/avatar"` : The actor's avatar from the `icon` property. Additional custom links can be added via WebFinger links dispatcher: `"http://ostatus.org/schema/1.0/subscribe"` : Remote follow endpoint (common in Mastodon). Uses a URI template with `{uri}` placeholder for the account being followed. Example WebFinger response (including both automatic and custom links): ~~~~ json { "subject": "acct:alice@your-domain.com", "links": [ { "rel": "self", "type": "application/activity+json", "href": "https://your-domain.com/users/alice" }, { "rel": "http://webfinger.net/rel/profile-page", "type": "text/html", "href": "https://your-domain.com/@alice" }, { "rel": "http://ostatus.org/schema/1.0/subscribe", "template": "https://your-domain.com/authorize_interaction?uri={uri}" } ] } ~~~~ [`Link`]: https://jsr.io/@fedify/fedify/doc/webfinger/~/Link [RFC 5988]: https://datatracker.ietf.org/doc/html/rfc5988 [RFC 6838]: https://datatracker.ietf.org/doc/html/rfc6838 [RFC 6570]: https://datatracker.ietf.org/doc/html/rfc6570 Customizing WebFinger endpoint ------------------------------ *This API is available since Fedify 1.9.0.* While Fedify automatically handles WebFinger responses for actors registered via `~Federatable.setActorDispatcher()`, you can add custom links using `~Federatable.setWebFingerLinksDispatcher()`. This is useful for adding non-standard links like Mastodon's remote follow endpoint: ~~~~ typescript twoslash // @noErrors: 2345 import { createFederation } from "@fedify/fedify"; const federation = createFederation({ // Omitted for brevity; see the related section for details. }); federation.setWebFingerLinksDispatcher(async (ctx, resource) => { return [ { rel: "http://ostatus.org/schema/1.0/subscribe", template: `https://your-domain.com/@${resource.pathname}/authorize_interaction?uri={uri}` } ]; } ); ~~~~ This gives results like below: ~~~~json { "subject": "acct:alice@your-domain.com", "links": [ { "rel": "self", "type": "application/activity+json", "href": "https://your-domain.com/users/alice" }, { "rel": "http://webfinger.net/rel/profile-page", "type": "text/html", "href": "https://your-domain.com/@alice" }, { "rel": "http://ostatus.org/schema/1.0/subscribe", "template": "https://your-domain.com/@alice@your-domain.com/authorize_interaction?uri={uri}" } ] } ~~~~ The WebFinger links dispatcher receives two parameters: `ctx` : The federation context `resource` : The URL queried via WebFinger > [!TIP] > The WebFinger endpoint is automatically exposed at `/.well-known/webfinger` > by the `Federation.fetch()` method. You don't need to manually handle this > route. > [!NOTE] > Before the introduction of `~Federatable.setWebFingerLinksDispatcher()` in > Fedify 1.9.0, WebFinger responses could only be customized through > `~Federatable.setActorDispatcher()` by setting the actor's `url` property. > This method still works and is sufficient for many use cases, though it > doesn't support the `template` field needed for features like Mastodon's > remote follow functionality. > > See the [WebFinger links section](./actor.md#webfinger-links) in the Actor > documentation for details on customizing WebFinger through actor properties. Looking up WebFinger -------------------- *This API is available since Fedify 1.6.0.* Use `~Context.lookupWebFinger()` to query remote WebFinger endpoints: ~~~~ typescript twoslash import { type Context } from "@fedify/fedify"; const ctx = null as unknown as Context<void>; // ---cut-before--- const webfingerData = await ctx.lookupWebFinger("acct:fedify@hollo.social"); ~~~~ If the lookup fails or the account doesn't exist, the method returns `null`. The returned WebFinger document contains links to various resources associated with the account, such as profile pages, ActivityPub actor URIs, and more: ~~~~ typescript twoslash import { type Context } from "@fedify/fedify"; const ctx = null as unknown as Context<void>; // ---cut-before--- const webfingerData = await ctx.lookupWebFinger("acct:fedify@hollo.social"); // Find the ActivityPub actor URI const activityPubActorLink = webfingerData?.links?.find(link => link.rel === "self" && link.type === "application/activity+json" ); if (activityPubActorLink?.href) { const actor = await ctx.lookupObject(activityPubActorLink.href); // Work with the actor... } ~~~~ > [!NOTE] > In most cases, you can use the higher-level `~Context.lookupObject()` method > which automatically performs WebFinger lookups when given a handle. > Use `~Context.lookupWebFinger()` when you need the raw WebFinger data or > want more direct control over the lookup process.
packages/fedify/src/federation/builder.ts +9 −0 Original line number Diff line number Diff line Loading @@ -22,6 +22,7 @@ import type { ObjectAuthorizePredicate, ObjectDispatcher, SharedInboxKeyDispatcher, WebFingerLinksDispatcher, } from "./callback.ts"; import type { Context, RequestContext } from "./context.ts"; import type { Loading @@ -48,6 +49,7 @@ export class FederationBuilderImpl<TContextData> router: Router; actorCallbacks?: ActorCallbacks<TContextData>; nodeInfoDispatcher?: NodeInfoDispatcher<TContextData>; webFingerLinksDispatcher?: WebFingerLinksDispatcher<TContextData>; objectCallbacks: Record<string, ObjectCallbacks<TContextData, string>>; objectTypeIds: Record< string, Loading Loading @@ -148,6 +150,7 @@ export class FederationBuilderImpl<TContextData> ? undefined : { ...this.actorCallbacks }; f.nodeInfoDispatcher = this.nodeInfoDispatcher; f.webFingerLinksDispatcher = this.webFingerLinksDispatcher; f.objectCallbacks = { ...this.objectCallbacks }; f.objectTypeIds = { ...this.objectTypeIds }; f.inboxPath = this.inboxPath; Loading Loading @@ -491,6 +494,12 @@ export class FederationBuilderImpl<TContextData> this.nodeInfoDispatcher = dispatcher; } setWebFingerLinksDispatcher( dispatcher: WebFingerLinksDispatcher<TContextData>, ): void { this.webFingerLinksDispatcher = dispatcher; } setObjectDispatcher<TObject extends Object, TParam extends string>( // deno-lint-ignore no-explicit-any cls: (new (...args: any[]) => TObject) & { typeId: URL }, Loading