181 lines
4.2 KiB
TypeScript
181 lines
4.2 KiB
TypeScript
import define from "../../define.js";
|
|
import { createPerson } from "@/remote/activitypub/models/person.js";
|
|
import { createNote } from "@/remote/activitypub/models/note.js";
|
|
import DbResolver from "@/remote/activitypub/db-resolver.js";
|
|
import Resolver from "@/remote/activitypub/resolver.js";
|
|
import { ApiError } from "../../error.js";
|
|
import { extractDbHost } from "@/misc/convert-host.js";
|
|
import { Users, Notes } from "@/models/index.js";
|
|
import type { Note } from "@/models/entities/note.js";
|
|
import type { CacheableLocalUser, User } from "@/models/entities/user.js";
|
|
import { isActor, isPost, getApId } from "@/remote/activitypub/type.js";
|
|
import type { SchemaType } from "@/misc/schema.js";
|
|
import { HOUR } from "@/const.js";
|
|
import { shouldBlockInstance } from "@/misc/should-block-instance.js";
|
|
import { updateQuestion } from "@/remote/activitypub/models/question.js";
|
|
import { populatePoll } from "@/models/repositories/note.js";
|
|
import { redisClient } from "@/db/redis.js";
|
|
|
|
export const meta = {
|
|
tags: ["federation"],
|
|
|
|
requireCredential: true,
|
|
|
|
limit: {
|
|
duration: HOUR,
|
|
max: 30,
|
|
},
|
|
|
|
errors: {
|
|
noSuchObject: {
|
|
message: "No such object.",
|
|
code: "NO_SUCH_OBJECT",
|
|
id: "dc94d745-1262-4e63-a17d-fecaa57efc82",
|
|
},
|
|
},
|
|
|
|
res: {
|
|
optional: false,
|
|
nullable: false,
|
|
oneOf: [
|
|
{
|
|
type: "object",
|
|
properties: {
|
|
type: {
|
|
type: "string",
|
|
optional: false,
|
|
nullable: false,
|
|
enum: ["User"],
|
|
},
|
|
object: {
|
|
type: "object",
|
|
optional: false,
|
|
nullable: false,
|
|
ref: "UserDetailedNotMe",
|
|
},
|
|
},
|
|
},
|
|
{
|
|
type: "object",
|
|
properties: {
|
|
type: {
|
|
type: "string",
|
|
optional: false,
|
|
nullable: false,
|
|
enum: ["Note"],
|
|
},
|
|
object: {
|
|
type: "object",
|
|
optional: false,
|
|
nullable: false,
|
|
ref: "Note",
|
|
},
|
|
},
|
|
},
|
|
],
|
|
},
|
|
} as const;
|
|
|
|
export const paramDef = {
|
|
type: "object",
|
|
properties: {
|
|
uri: { type: "string" },
|
|
},
|
|
required: ["uri"],
|
|
} as const;
|
|
|
|
export default define(meta, paramDef, async (ps, me) => {
|
|
const object = await fetchAny(ps.uri, me);
|
|
if (object) {
|
|
return object;
|
|
} else {
|
|
throw new ApiError(meta.errors.noSuchObject);
|
|
}
|
|
});
|
|
|
|
/***
|
|
* Resolve User or Note from URI
|
|
*/
|
|
async function fetchAny(
|
|
uri: string,
|
|
me: CacheableLocalUser | null | undefined,
|
|
): Promise<SchemaType<typeof meta["res"]> | null> {
|
|
// Wait if blocked.
|
|
if (await shouldBlockInstance(extractDbHost(uri))) return null;
|
|
|
|
const dbResolver = new DbResolver();
|
|
|
|
const [user, note] = await Promise.all([
|
|
dbResolver.getUserFromApId(uri),
|
|
dbResolver.getNoteFromApId(uri),
|
|
]);
|
|
let local = await mergePack(me, user, note);
|
|
if (local) {
|
|
if (local.type === "Note" && note?.uri && note.hasPoll) {
|
|
// Update questions if the stored (remote) note contains the poll
|
|
const key = `pollFetched:${note.uri}`;
|
|
if ((await redisClient.exists(key)) === 0) {
|
|
if (await updateQuestion(note.uri)) {
|
|
local.object.poll = await populatePoll(note, me?.id ?? null);
|
|
}
|
|
// Allow fetching the poll again after 1 minute
|
|
await redisClient.set(key, 1, "EX", 60);
|
|
}
|
|
}
|
|
return local;
|
|
}
|
|
|
|
// fetching Object once from remote
|
|
const resolver = new Resolver();
|
|
resolver.setUser(me);
|
|
const object = await resolver.resolve(uri);
|
|
|
|
// /@user If a URI other than the id is specified,
|
|
// the URI is determined here
|
|
if (uri !== object.id) {
|
|
local = await mergePack(
|
|
me,
|
|
...(await Promise.all([
|
|
dbResolver.getUserFromApId(getApId(object)),
|
|
dbResolver.getNoteFromApId(getApId(object)),
|
|
])),
|
|
);
|
|
if (local != null) return local;
|
|
}
|
|
|
|
return await mergePack(
|
|
me,
|
|
isActor(object)
|
|
? await createPerson(getApId(object), resolver.reset())
|
|
: null,
|
|
isPost(object)
|
|
? await createNote(getApId(object), resolver.reset(), true)
|
|
: null,
|
|
);
|
|
}
|
|
|
|
async function mergePack(
|
|
me: CacheableLocalUser | null | undefined,
|
|
user: User | null | undefined,
|
|
note: Note | null | undefined,
|
|
): Promise<SchemaType<typeof meta.res> | null> {
|
|
if (user != null) {
|
|
return {
|
|
type: "User",
|
|
object: await Users.pack(user, me, { detail: true }),
|
|
};
|
|
} else if (note != null) {
|
|
try {
|
|
const object = await Notes.pack(note, me, { detail: true });
|
|
|
|
return {
|
|
type: "Note",
|
|
object,
|
|
};
|
|
} catch (e) {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|