import { IsNull } from "typeorm"; import follow from "@/services/following/create.js"; import * as Post from "@/misc/post.js"; import create from "@/services/note/create.js"; import { downloadTextFile } from "@/misc/download-text-file.js"; import { Users, DriveFiles } from "@/models/index.js"; import type { DbUserImportPostsJobData } from "@/queue/types.js"; import { queueLogger } from "../../logger.js"; import type Bull from "bull"; import { htmlToMfm } from "@/remote/activitypub/misc/html-to-mfm.js"; import { resolveNote } from "@/remote/activitypub/models/note.js"; import { Note } from "@/models/entities/note.js"; const logger = queueLogger.createSubLogger("import-posts"); export async function importPosts( job: Bull.Job, done: any, ): Promise { logger.info(`Importing posts of ${job.data.user.id} ...`); const user = await Users.findOneBy({ id: job.data.user.id }); if (user == null) { done(); return; } const file = await DriveFiles.findOneBy({ id: job.data.fileId, }); if (file == null) { done(); return; } const json = await downloadTextFile(file.url); let linenum = 0; try { const parsed = JSON.parse(json); if (parsed instanceof Array) { logger.info("Parsing key style posts"); for (const post of JSON.parse(json)) { try { linenum++; if (post.replyId != null) { continue; } if (post.renoteId != null) { continue; } if (post.visibility !== "public") { continue; } const { text, cw, localOnly, createdAt } = Post.parse(post); logger.info(`Posting[${linenum}] ...`); const note = await create(user, { createdAt: createdAt, files: undefined, poll: undefined, text: text || undefined, reply: null, renote: null, cw: cw, localOnly, visibility: "hidden", visibleUsers: [], channel: null, apMentions: new Array(0), apHashtags: undefined, apEmojis: undefined, }); } catch (e) { logger.warn(`Error in line:${linenum} ${e}`); } } } else if (parsed instanceof Object) { logger.info("Parsing animal style posts"); for (const post of parsed.orderedItems) { async () => { try { linenum++; let reply: Note | null = null; if (post.object.inReplyTo != null) { reply = await resolveNote(post.object.inReplyTo); } if (post.directMessage) { return; } if (job.data.signatureCheck) { if (!post.signature) { return; } } let text; try { text = htmlToMfm(post.object.content, post.object.tag); } catch (e) { return; } logger.info(`Posting[${linenum}] ...`); const note = await create(user, { createdAt: new Date(post.object.published), files: undefined, poll: undefined, text: text || undefined, reply, renote: null, cw: post.sensitive, localOnly: false, visibility: "hidden", visibleUsers: [], channel: null, apMentions: new Array(0), apHashtags: undefined, apEmojis: undefined, }); } catch (e) { logger.warn(`Error in line:${linenum} ${e}`); } }; } } } catch (e) { // handle error logger.warn(`Error reading: ${e}`); } logger.succ("Imported"); done(); }