import all misskey self note thread and allow import multiple times to restore missing parts
This commit is contained in:
parent
90724dc06c
commit
109ed0aa80
7 changed files with 157 additions and 35 deletions
|
@ -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 || ""),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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,
|
||||||
|
userId: user.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
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,
|
poll: undefined,
|
||||||
text: text || undefined,
|
text: text || undefined,
|
||||||
reply: null,
|
reply: post.replyId ? job.data.parent : null,
|
||||||
renote: null,
|
renote: post.renoteId ? job.data.parent : null,
|
||||||
cw: cw,
|
cw: cw,
|
||||||
localOnly,
|
localOnly,
|
||||||
visibility: "hidden",
|
visibility: visibility,
|
||||||
visibleUsers: [],
|
visibleUsers: [],
|
||||||
channel: null,
|
channel: null,
|
||||||
apMentions: new Array(0),
|
apMentions: new Array(0),
|
||||||
apHashtags: undefined,
|
apHashtags: undefined,
|
||||||
apEmojis: 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();
|
||||||
}
|
}
|
||||||
|
|
|
@ -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,8 +70,28 @@ export async function importMastoPost(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
let note = await Notes.findOneBy({
|
||||||
|
createdAt: new Date(post.object.published),
|
||||||
|
text: text,
|
||||||
|
userId: user.id,
|
||||||
|
});
|
||||||
|
|
||||||
const note = await create(user, {
|
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),
|
createdAt: new Date(post.object.published),
|
||||||
files: files.length == 0 ? undefined : files,
|
files: files.length == 0 ? undefined : files,
|
||||||
poll: undefined,
|
poll: undefined,
|
||||||
|
@ -77,13 +100,17 @@ export async function importMastoPost(
|
||||||
renote: null,
|
renote: null,
|
||||||
cw: post.object.sensitive ? post.object.summary : undefined,
|
cw: post.object.sensitive ? post.object.summary : undefined,
|
||||||
localOnly: false,
|
localOnly: false,
|
||||||
visibility: "hidden",
|
visibility: "hiddenpublic",
|
||||||
visibleUsers: [],
|
visibleUsers: [],
|
||||||
channel: null,
|
channel: null,
|
||||||
apMentions: new Array(0),
|
apMentions: new Array(0),
|
||||||
apHashtags: undefined,
|
apHashtags: undefined,
|
||||||
apEmojis: undefined,
|
apEmojis: undefined,
|
||||||
});
|
});
|
||||||
|
logger.info(`Create new note`);
|
||||||
|
} else {
|
||||||
|
logger.info(`Note exist`);
|
||||||
|
}
|
||||||
job.progress(100);
|
job.progress(100);
|
||||||
done();
|
done();
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
|
}
|
||||||
|
|
|
@ -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 =
|
||||||
|
|
|
@ -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 (
|
||||||
|
|
Loading…
Reference in a new issue