Merge branch 'develop' of git.joinfirefish.org:firefish/firefish into develop

This commit is contained in:
ThatOneCalculator 2023-07-31 18:23:15 -07:00
commit 800ba2dc4f
No known key found for this signature in database
GPG Key ID: 8703CACD01000000
8 changed files with 158 additions and 36 deletions

View File

@ -1128,7 +1128,7 @@ _mfm:
bold: "Fett" bold: "Fett"
boldDescription: "Zeichen zur Betonung dicker erscheinen lassen." boldDescription: "Zeichen zur Betonung dicker erscheinen lassen."
small: "Klein" small: "Klein"
smallDescription: "Zeigt den Inhalt klein und dünn an" smallDescription: "Inhalt klein und dünn anzeigen."
center: "Zentrieren" center: "Zentrieren"
centerDescription: "Inhalt zentriert anzeigen." centerDescription: "Inhalt zentriert anzeigen."
inlineCode: "Code (Eingebettet)" inlineCode: "Code (Eingebettet)"

View File

@ -1,16 +1,18 @@
export type Post = { export type Post = {
text: string | null; text: string | undefined;
cw: string | null; cw: string | null;
localOnly: boolean; localOnly: boolean;
createdAt: Date; createdAt: Date;
visibility: string;
}; };
export function parse(acct: any): Post { export function parse(acct: any): Post {
return { return {
text: acct.text, text: acct.text || undefined,
cw: acct.cw, cw: acct.cw,
localOnly: acct.localOnly, localOnly: acct.localOnly,
createdAt: new Date(acct.createdAt), createdAt: new Date(acct.createdAt),
visibility: "hidden" + (acct.visibility || ""),
}; };
} }

View File

@ -28,6 +28,7 @@ import {
backgroundQueue, backgroundQueue,
} from "./queues.js"; } from "./queues.js";
import type { ThinUser } from "./types.js"; import type { ThinUser } from "./types.js";
import { Note } from "@/models/entities/note.js";
function renderError(e: Error): any { function renderError(e: Error): any {
return { return {
@ -358,6 +359,7 @@ export function createImportCkPostJob(
user: ThinUser, user: ThinUser,
post: any, post: any,
signatureCheck: boolean, signatureCheck: boolean,
parent: Note | null = null,
) { ) {
return dbQueue.add( return dbQueue.add(
"importCkPost", "importCkPost",
@ -365,6 +367,7 @@ export function createImportCkPostJob(
user: user, user: user,
post: post, post: post,
signatureCheck: signatureCheck, signatureCheck: signatureCheck,
parent: parent,
}, },
{ {
removeOnComplete: true, removeOnComplete: true,

View File

@ -3,7 +3,13 @@ import create from "@/services/note/create.js";
import { Users } from "@/models/index.js"; import { Users } from "@/models/index.js";
import type { DbUserImportMastoPostJobData } from "@/queue/types.js"; import type { DbUserImportMastoPostJobData } from "@/queue/types.js";
import { queueLogger } from "../../logger.js"; import { queueLogger } from "../../logger.js";
import { uploadFromUrl } from "@/services/drive/upload-from-url.js";
import type { DriveFile } from "@/models/entities/drive-file.js";
import type Bull from "bull"; import type Bull from "bull";
import { createImportCkPostJob } from "@/queue/index.js";
import { Notes, NoteEdits } from "@/models/index.js";
import type { Note } from "@/models/entities/note.js";
import { genId } from "@/misc/gen-id.js";
const logger = queueLogger.createSubLogger("import-firefish-post"); const logger = queueLogger.createSubLogger("import-firefish-post");
@ -17,6 +23,7 @@ export async function importCkPost(
return; return;
} }
const post = job.data.post; const post = job.data.post;
/*
if (post.replyId != null) { if (post.replyId != null) {
done(); done();
return; return;
@ -29,23 +36,74 @@ export async function importCkPost(
done(); done();
return; return;
} }
const { text, cw, localOnly, createdAt } = Post.parse(post); */
const note = await create(user, { const urls = (post.files || [])
.map((x: any) => x.url)
.filter((x: String) => x.startsWith("http"));
const files: DriveFile[] = [];
for (const url of urls) {
try {
const file = await uploadFromUrl({
url: url,
user: user,
});
files.push(file);
} catch (e) {
logger.error(`Skipped adding file to drive: ${url}`);
}
}
const { text, cw, localOnly, createdAt, visibility } = Post.parse(post);
let note = await Notes.findOneBy({
createdAt: createdAt, createdAt: createdAt,
files: undefined, text: text,
poll: undefined, userId: user.id,
text: text || undefined,
reply: null,
renote: null,
cw: cw,
localOnly,
visibility: "hidden",
visibleUsers: [],
channel: null,
apMentions: new Array(0),
apHashtags: undefined,
apEmojis: undefined,
}); });
if (note && (note?.fileIds?.length || 0) < files.length) {
const update: Partial<Note> = {};
update.fileIds = files.map((x) => x.id);
await Notes.update(note.id, update);
await NoteEdits.insert({
id: genId(),
noteId: note.id,
text: note.text || undefined,
cw: note.cw,
fileIds: note.fileIds,
updatedAt: new Date(),
});
logger.info(`Note file updated`);
}
if (!note) {
note = await create(user, {
createdAt: createdAt,
files: files.length == 0 ? undefined : files,
poll: undefined,
text: text || undefined,
reply: post.replyId ? job.data.parent : null,
renote: post.renoteId ? job.data.parent : null,
cw: cw,
localOnly,
visibility: visibility,
visibleUsers: [],
channel: null,
apMentions: new Array(0),
apHashtags: undefined,
apEmojis: undefined,
});
logger.info(`Create new note`);
} else {
logger.info(`Note exist`);
}
logger.succ("Imported"); logger.succ("Imported");
if (post.childNotes) {
for (const child of post.childNotes) {
createImportCkPostJob(
job.data.user,
child,
job.data.signatureCheck,
note,
);
}
}
done(); done();
} }

View File

@ -8,6 +8,9 @@ import { resolveNote } from "@/remote/activitypub/models/note.js";
import { Note } from "@/models/entities/note.js"; import { Note } from "@/models/entities/note.js";
import { uploadFromUrl } from "@/services/drive/upload-from-url.js"; import { uploadFromUrl } from "@/services/drive/upload-from-url.js";
import type { DriveFile } from "@/models/entities/drive-file.js"; import type { DriveFile } from "@/models/entities/drive-file.js";
import { Notes, NoteEdits } from "@/models/index.js";
import type { Note } from "@/models/entities/note.js";
import { genId } from "@/misc/gen-id.js";
const logger = queueLogger.createSubLogger("import-masto-post"); const logger = queueLogger.createSubLogger("import-masto-post");
@ -67,23 +70,47 @@ export async function importMastoPost(
} }
} }
} }
let note = await Notes.findOneBy({
const note = await create(user, {
createdAt: new Date(post.object.published), createdAt: new Date(post.object.published),
files: files.length == 0 ? undefined : files, text: text,
poll: undefined, userId: user.id,
text: text || undefined,
reply,
renote: null,
cw: post.object.sensitive ? post.object.summary : undefined,
localOnly: false,
visibility: "hidden",
visibleUsers: [],
channel: null,
apMentions: new Array(0),
apHashtags: undefined,
apEmojis: undefined,
}); });
if (note && (note?.fileIds?.length || 0) < files.length) {
const update: Partial<Note> = {};
update.fileIds = files.map((x) => x.id);
await Notes.update(note.id, update);
await NoteEdits.insert({
id: genId(),
noteId: note.id,
text: note.text || undefined,
cw: note.cw,
fileIds: note.fileIds,
updatedAt: new Date(),
});
logger.info(`Note file updated`);
}
if (!note) {
note = await create(user, {
createdAt: new Date(post.object.published),
files: files.length == 0 ? undefined : files,
poll: undefined,
text: text || undefined,
reply,
renote: null,
cw: post.object.sensitive ? post.object.summary : undefined,
localOnly: false,
visibility: "hiddenpublic",
visibleUsers: [],
channel: null,
apMentions: new Array(0),
apHashtags: undefined,
apEmojis: undefined,
});
logger.info(`Create new note`);
} else {
logger.info(`Note exist`);
}
job.progress(100); job.progress(100);
done(); done();

View File

@ -57,7 +57,8 @@ export async function importPosts(
const parsed = JSON.parse(json); const parsed = JSON.parse(json);
if (parsed instanceof Array) { if (parsed instanceof Array) {
logger.info("Parsing key style posts"); logger.info("Parsing key style posts");
for (const post of JSON.parse(json)) { const arr = recreateChain(parsed);
for (const post of arr) {
createImportCkPostJob(job.data.user, post, job.data.signatureCheck); createImportCkPostJob(job.data.user, post, job.data.signatureCheck);
} }
} else if (parsed instanceof Object) { } else if (parsed instanceof Object) {
@ -74,3 +75,32 @@ export async function importPosts(
logger.succ("Imported"); logger.succ("Imported");
done(); done();
} }
function recreateChain(arr: any[]): any {
type NotesMap = {
[id: string]: any;
};
const notesTree: any[] = [];
const lookup: NotesMap = {};
for (const note of arr) {
lookup[`${note.id}`] = note;
note.childNotes = [];
if (note.replyId == null && note.renoteId == null) {
notesTree.push(note);
}
}
for (const note of arr) {
let parent = null;
if (note.replyId != null) {
parent = lookup[`${note.replyId}`];
}
if (note.renoteId != null) {
parent = lookup[`${note.renoteId}`];
}
if (parent) {
parent.childNotes.push(note);
}
}
return notesTree;
}

View File

@ -52,6 +52,7 @@ export type DbUserImportMastoPostJobData = {
user: ThinUser; user: ThinUser;
post: any; post: any;
signatureCheck: boolean; signatureCheck: boolean;
parent: Note | null;
}; };
export type ObjectStorageJobData = export type ObjectStorageJobData =

View File

@ -172,7 +172,7 @@ export default async (
// rome-ignore lint/suspicious/noAsyncPromiseExecutor: FIXME // rome-ignore lint/suspicious/noAsyncPromiseExecutor: FIXME
new Promise<Note>(async (res, rej) => { new Promise<Note>(async (res, rej) => {
const dontFederateInitially = const dontFederateInitially =
data.localOnly || data.visibility === "hidden"; data.localOnly || data.visibility?.startsWith("hidden");
// If you reply outside the channel, match the scope of the target. // If you reply outside the channel, match the scope of the target.
// TODO (I think it's a process that could be done on the client side, but it's server side for now.) // TODO (I think it's a process that could be done on the client side, but it's server side for now.)
@ -206,7 +206,8 @@ export default async (
if (data.channel != null) data.visibility = "public"; if (data.channel != null) data.visibility = "public";
if (data.channel != null) data.visibleUsers = []; if (data.channel != null) data.visibleUsers = [];
if (data.channel != null) data.localOnly = true; if (data.channel != null) data.localOnly = true;
if (data.visibility === "hidden") data.visibility = "public"; if (data.visibility.startsWith("hidden"))
data.visibility = data.visibility.slice(6);
// enforce silent clients on server // enforce silent clients on server
if ( if (