mirror of
https://iceshrimp.dev/limepotato/jormungandr-bite.git
synced 2024-11-22 01:47:39 -07:00
[mastodon-client] Use new backend service for user (profile) updates
This fixes profile updates not immediately federating when edited through the Mastodon client API.
This commit is contained in:
parent
d42a1eeb63
commit
44fb31ab13
3 changed files with 81 additions and 70 deletions
|
@ -2,7 +2,7 @@ import RE2 from "re2";
|
||||||
import * as mfm from "mfm-js";
|
import * as mfm from "mfm-js";
|
||||||
import { publishMainStream, publishUserEvent } from "@/services/stream.js";
|
import { publishMainStream, publishUserEvent } from "@/services/stream.js";
|
||||||
import acceptAllFollowRequests from "@/services/following/requests/accept-all.js";
|
import acceptAllFollowRequests from "@/services/following/requests/accept-all.js";
|
||||||
import { publishToFollowers } from "@/services/i/update.js";
|
import { publishToFollowers, updateUserProfileData } from "@/services/i/update.js";
|
||||||
import { extractCustomEmojisFromMfm } from "@/misc/extract-custom-emojis-from-mfm.js";
|
import { extractCustomEmojisFromMfm } from "@/misc/extract-custom-emojis-from-mfm.js";
|
||||||
import { extractHashtags } from "@/misc/extract-hashtags.js";
|
import { extractHashtags } from "@/misc/extract-hashtags.js";
|
||||||
import { updateUsertags } from "@/services/update-hashtag.js";
|
import { updateUsertags } from "@/services/update-hashtag.js";
|
||||||
|
@ -274,65 +274,5 @@ export default define(meta, paramDef, async (ps, _user, token) => {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
//#region emojis/tags
|
return updateUserProfileData(user, profile, updates, profileUpdates, isSecure);
|
||||||
|
|
||||||
let emojis = [] as string[];
|
|
||||||
let tags = [] as string[];
|
|
||||||
|
|
||||||
const newName = updates.name === undefined ? user.name : updates.name;
|
|
||||||
const newDescription =
|
|
||||||
profileUpdates.description === undefined
|
|
||||||
? profile.description
|
|
||||||
: profileUpdates.description;
|
|
||||||
|
|
||||||
if (newName != null) {
|
|
||||||
const tokens = mfm.parseSimple(newName);
|
|
||||||
emojis = emojis.concat(extractCustomEmojisFromMfm(tokens!));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (newDescription != null) {
|
|
||||||
const tokens = mfm.parse(newDescription);
|
|
||||||
emojis = emojis.concat(extractCustomEmojisFromMfm(tokens!));
|
|
||||||
tags = extractHashtags(tokens!)
|
|
||||||
.map((tag) => normalizeForSearch(tag))
|
|
||||||
.splice(0, 32);
|
|
||||||
}
|
|
||||||
|
|
||||||
updates.emojis = emojis;
|
|
||||||
updates.tags = tags;
|
|
||||||
|
|
||||||
// ハッシュタグ更新
|
|
||||||
updateUsertags(user, tags);
|
|
||||||
//#endregion
|
|
||||||
|
|
||||||
if (Object.keys(updates).length > 0) await Users.update(user.id, updates);
|
|
||||||
if (Object.keys(profileUpdates).length > 0) {
|
|
||||||
await UserProfiles.update(user.id, profileUpdates);
|
|
||||||
await UserProfiles.updateMentions(user.id);
|
|
||||||
}
|
|
||||||
|
|
||||||
const iObj = await Users.pack<true, true>(user.id, user, {
|
|
||||||
detail: true,
|
|
||||||
includeSecrets: isSecure,
|
|
||||||
});
|
|
||||||
|
|
||||||
// Publish meUpdated event
|
|
||||||
publishMainStream(user.id, "meUpdated", iObj);
|
|
||||||
publishUserEvent(
|
|
||||||
user.id,
|
|
||||||
"updateUserProfile",
|
|
||||||
await UserProfiles.findOneBy({ userId: user.id }),
|
|
||||||
);
|
|
||||||
|
|
||||||
// 鍵垢を解除したとき、溜まっていたフォローリクエストがあるならすべて承認
|
|
||||||
if (user.isLocked && ps.isLocked === false) {
|
|
||||||
acceptAllFollowRequests(user);
|
|
||||||
}
|
|
||||||
|
|
||||||
// フォロワーにUpdateを配信
|
|
||||||
UserProfiles.updateMentions(user.id).finally(() => {
|
|
||||||
publishToFollowers(user.id);
|
|
||||||
});
|
|
||||||
|
|
||||||
return iObj;
|
|
||||||
});
|
});
|
||||||
|
|
|
@ -44,6 +44,7 @@ import { MastoContext } from "@/server/api/mastodon/index.js";
|
||||||
import { resolveUser } from "@/remote/resolve-user.js";
|
import { resolveUser } from "@/remote/resolve-user.js";
|
||||||
import { updatePerson } from "@/remote/activitypub/models/person.js";
|
import { updatePerson } from "@/remote/activitypub/models/person.js";
|
||||||
import { promiseEarlyReturn } from "@/prelude/promise.js";
|
import { promiseEarlyReturn } from "@/prelude/promise.js";
|
||||||
|
import { updateUserProfileData } from "@/services/i/update.js";
|
||||||
|
|
||||||
export type AccountCache = {
|
export type AccountCache = {
|
||||||
locks: AsyncLock;
|
locks: AsyncLock;
|
||||||
|
@ -181,12 +182,18 @@ export class UserHelpers {
|
||||||
|
|
||||||
if (formData.fields_attributes) {
|
if (formData.fields_attributes) {
|
||||||
profileUpdates.fields = await Promise.all(formData.fields_attributes.map(async field => {
|
profileUpdates.fields = await Promise.all(formData.fields_attributes.map(async field => {
|
||||||
const verified = field.value.startsWith("http") ? await verifyLink(field.value, user.username) : undefined;
|
if (!(field.name.trim() === "" && field.value.trim() === "")) {
|
||||||
|
if (field.name.trim() === "") throw new MastoApiError(400, "Field name can not be empty");
|
||||||
|
if (field.value.trim() === "") throw new MastoApiError(400, "Field value can not be empty");
|
||||||
|
}
|
||||||
|
const verified = field.value.startsWith("http")
|
||||||
|
? (await promiseEarlyReturn(verifyLink(field.value, user.username), 1500)) ?? false
|
||||||
|
: undefined;
|
||||||
return {
|
return {
|
||||||
...field,
|
...field,
|
||||||
verified
|
verified
|
||||||
};
|
};
|
||||||
}));
|
})).then(p => p.filter(field => field.name.trim().length > 0 && field.value.length > 0));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (formData.display_name) updates.name = formData.display_name;
|
if (formData.display_name) updates.name = formData.display_name;
|
||||||
|
@ -195,11 +202,7 @@ export class UserHelpers {
|
||||||
if (formData.bot) updates.isBot = formData.bot;
|
if (formData.bot) updates.isBot = formData.bot;
|
||||||
if (formData.discoverable) updates.isExplorable = formData.discoverable;
|
if (formData.discoverable) updates.isExplorable = formData.discoverable;
|
||||||
|
|
||||||
if (Object.keys(updates).length > 0) await Users.update(user.id, updates);
|
await updateUserProfileData(user, null, updates, profileUpdates, false);
|
||||||
if (Object.keys(profileUpdates).length > 0) {
|
|
||||||
await UserProfiles.update({ userId: user.id }, profileUpdates);
|
|
||||||
await promiseEarlyReturn(UserProfiles.updateMentions(user.id), 1500);
|
|
||||||
}
|
|
||||||
|
|
||||||
return this.verifyCredentials(ctx);
|
return this.verifyCredentials(ctx);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,10 +1,78 @@
|
||||||
import renderUpdate from "@/remote/activitypub/renderer/update.js";
|
import renderUpdate from "@/remote/activitypub/renderer/update.js";
|
||||||
import { renderActivity } from "@/remote/activitypub/renderer/index.js";
|
import { renderActivity } from "@/remote/activitypub/renderer/index.js";
|
||||||
import { Users } from "@/models/index.js";
|
import { UserProfiles, Users } from "@/models/index.js";
|
||||||
import type { User } from "@/models/entities/user.js";
|
import type { User } from "@/models/entities/user.js";
|
||||||
import { renderPerson } from "@/remote/activitypub/renderer/person.js";
|
import { renderPerson } from "@/remote/activitypub/renderer/person.js";
|
||||||
import { deliverToFollowers } from "@/remote/activitypub/deliver-manager.js";
|
import { deliverToFollowers } from "@/remote/activitypub/deliver-manager.js";
|
||||||
import { deliverToRelays } from "../relay.js";
|
import { deliverToRelays } from "../relay.js";
|
||||||
|
import { extractCustomEmojisFromMfm } from "@/misc/extract-custom-emojis-from-mfm.js";
|
||||||
|
import { extractHashtags } from "@/misc/extract-hashtags.js";
|
||||||
|
import { normalizeForSearch } from "@/misc/normalize-for-search.js";
|
||||||
|
import { updateUsertags } from "@/services/update-hashtag.js";
|
||||||
|
import { publishMainStream, publishUserEvent } from "@/services/stream.js";
|
||||||
|
import acceptAllFollowRequests from "@/services/following/requests/accept-all.js";
|
||||||
|
import { UserProfile } from "@/models/entities/user-profile.js";
|
||||||
|
import mfm from "mfm-js";
|
||||||
|
import { promiseEarlyReturn } from "@/prelude/promise.js";
|
||||||
|
|
||||||
|
export async function updateUserProfileData(user: User, profile: UserProfile | null, updates: Partial<User>, profileUpdates: Partial<UserProfile>, isSecure: boolean) {
|
||||||
|
if (!profile) profile = await UserProfiles.findOneByOrFail({ userId: user.id });
|
||||||
|
|
||||||
|
let emojis = [] as string[];
|
||||||
|
let tags = [] as string[];
|
||||||
|
|
||||||
|
const newName = updates.name === undefined ? user.name : updates.name;
|
||||||
|
const newDescription =
|
||||||
|
profileUpdates.description === undefined
|
||||||
|
? profile.description
|
||||||
|
: profileUpdates.description;
|
||||||
|
|
||||||
|
if (newName != null) {
|
||||||
|
const tokens = mfm.parseSimple(newName);
|
||||||
|
emojis = emojis.concat(extractCustomEmojisFromMfm(tokens!));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (newDescription != null) {
|
||||||
|
const tokens = mfm.parse(newDescription);
|
||||||
|
emojis = emojis.concat(extractCustomEmojisFromMfm(tokens!));
|
||||||
|
tags = extractHashtags(tokens!)
|
||||||
|
.map((tag) => normalizeForSearch(tag))
|
||||||
|
.splice(0, 32);
|
||||||
|
}
|
||||||
|
|
||||||
|
updates.emojis = emojis;
|
||||||
|
updates.tags = tags;
|
||||||
|
|
||||||
|
updateUsertags(user, tags);
|
||||||
|
|
||||||
|
if (Object.keys(updates).length > 0) await Users.update(user.id, updates);
|
||||||
|
if (Object.keys(profileUpdates).length > 0) {
|
||||||
|
await UserProfiles.update(user.id, profileUpdates);
|
||||||
|
await promiseEarlyReturn(UserProfiles.updateMentions(user.id), 1500);
|
||||||
|
}
|
||||||
|
|
||||||
|
const iObj = await Users.pack<true, true>(user.id, user, {
|
||||||
|
detail: true,
|
||||||
|
includeSecrets: isSecure,
|
||||||
|
});
|
||||||
|
|
||||||
|
publishMainStream(user.id, "meUpdated", iObj);
|
||||||
|
publishUserEvent(
|
||||||
|
user.id,
|
||||||
|
"updateUserProfile",
|
||||||
|
await UserProfiles.findOneByOrFail({ userId: user.id }),
|
||||||
|
);
|
||||||
|
|
||||||
|
if (user.isLocked && updates.isLocked === false) {
|
||||||
|
acceptAllFollowRequests(user);
|
||||||
|
}
|
||||||
|
|
||||||
|
UserProfiles.updateMentions(user.id).finally(() => {
|
||||||
|
publishToFollowers(user.id);
|
||||||
|
});
|
||||||
|
|
||||||
|
return iObj;
|
||||||
|
}
|
||||||
|
|
||||||
export async function publishToFollowers(userId: User["id"]) {
|
export async function publishToFollowers(userId: User["id"]) {
|
||||||
const user = await Users.findOneBy({ id: userId });
|
const user = await Users.findOneBy({ id: userId });
|
||||||
|
|
Loading…
Reference in a new issue