server/classes/functions/federation.ts

74 lines
2.1 KiB
TypeScript
Raw Normal View History

import { debugRequest } from "@/api";
import { getLogger } from "@logtape/logtape";
import { SignatureConstructor } from "@lysand-org/federation";
import type { Entity, Undo } from "@lysand-org/federation/types";
2024-04-10 07:51:00 +02:00
import { config } from "config-manager";
import type { User } from "~/packages/database-interface/user";
2024-04-10 07:51:00 +02:00
export const localObjectUri = (id: string) =>
new URL(`/objects/${id}`, config.http.base_url).toString();
2024-04-10 07:51:00 +02:00
export const objectToInboxRequest = async (
object: Entity,
2024-04-10 07:51:00 +02:00
author: User,
userToSendTo: User,
): Promise<Request> => {
if (userToSendTo.isLocal() || !userToSendTo.data.endpoints?.inbox) {
throw new Error("UserToSendTo has no inbox or is a local user");
}
if (author.isRemote()) {
throw new Error("Author is a remote user");
2024-04-10 07:51:00 +02:00
}
const privateKey = await crypto.subtle.importKey(
"pkcs8",
Buffer.from(author.data.privateKey ?? "", "base64"),
2024-04-10 07:51:00 +02:00
"Ed25519",
false,
["sign"],
);
const ctor = new SignatureConstructor(privateKey, author.getUri());
2024-04-10 07:51:00 +02:00
const userInbox = new URL(userToSendTo.data.endpoints?.inbox ?? "");
2024-04-10 07:51:00 +02:00
const request = new Request(userInbox, {
2024-04-10 07:51:00 +02:00
method: "POST",
headers: {
"Content-Type": "application/json",
Origin: new URL(config.http.base_url).host,
2024-04-10 07:51:00 +02:00
},
body: JSON.stringify(object),
});
const { request: signed, signedString } = await ctor.sign(request);
if (config.debug.federation) {
// Debug request
await debugRequest(signed);
const logger = getLogger("federation");
// Log public key
logger.debug`Sender public key: ${author.data.publicKey}`;
// Log signed string
logger.debug`Signed string:\n${signedString}`;
}
return signed;
2024-04-10 07:51:00 +02:00
};
export const undoFederationRequest = (undoer: User, uri: string): Undo => {
const id = crypto.randomUUID();
return {
type: "Undo",
id,
author: undoer.getUri(),
created_at: new Date().toISOString(),
object: uri,
uri: new URL(`/undos/${id}`, config.http.base_url).toString(),
};
};