server/database/entities/Federation.ts

85 lines
2.3 KiB
TypeScript
Raw Normal View History

import { debugRequest } from "@/api";
import {
type EntityValidator,
SignatureConstructor,
} from "@lysand-org/federation";
2024-04-10 07:51:00 +02:00
import { config } from "config-manager";
import type { User } from "~/packages/database-interface/user";
import { LogLevel, LogManager } from "~/packages/log-manager";
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: typeof EntityValidator.$Entity,
2024-04-10 07:51:00 +02:00
author: User,
userToSendTo: User,
): Promise<Request> => {
if (userToSendTo.isLocal() || !userToSendTo.getUser().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.getUser().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.getUser().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);
// Log public key
new LogManager(Bun.stdout).log(
LogLevel.DEBUG,
"Inbox.Signature",
`Sender public key: ${author.getUser().publicKey}`,
);
// Log signed string
new LogManager(Bun.stdout).log(
LogLevel.DEBUG,
"Inbox.Signature",
`Signed string:\n${signedString}`,
);
}
return signed;
2024-04-10 07:51:00 +02:00
};
export const undoFederationRequest = (
undoer: User,
uri: string,
): typeof EntityValidator.$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(),
};
};