pleroma/app/soapbox/schemas/status.ts

151 lines
5 KiB
TypeScript
Raw Normal View History

2023-05-11 09:56:19 -07:00
import escapeTextContentForBrowser from 'escape-html';
2023-04-10 13:22:08 -07:00
import { z } from 'zod';
2023-05-11 09:56:19 -07:00
import emojify from 'soapbox/features/emoji';
2023-05-18 15:09:15 -07:00
import { stripCompatibilityFeatures, unescapeHTML } from 'soapbox/utils/html';
2023-05-11 09:56:19 -07:00
2023-05-04 10:13:39 -07:00
import { accountSchema } from './account';
import { attachmentSchema } from './attachment';
import { cardSchema } from './card';
import { customEmojiSchema } from './custom-emoji';
import { emojiReactionSchema } from './emoji-reaction';
2023-05-18 15:09:15 -07:00
import { eventSchema } from './event';
2023-05-04 10:13:39 -07:00
import { groupSchema } from './group';
import { mentionSchema } from './mention';
import { pollSchema } from './poll';
import { tagSchema } from './tag';
2023-05-11 09:56:19 -07:00
import { contentSchema, dateSchema, filteredArray, makeCustomEmojiMap } from './utils';
2023-04-10 13:22:08 -07:00
2023-06-13 20:12:42 -07:00
import type { Resolve } from 'soapbox/utils/types';
const statusPleromaSchema = z.object({
emoji_reactions: filteredArray(emojiReactionSchema),
event: eventSchema.nullish().catch(undefined),
quote: z.literal(null).catch(null),
quote_visible: z.boolean().catch(true),
});
2023-05-04 10:13:39 -07:00
const baseStatusSchema = z.object({
account: accountSchema,
application: z.object({
name: z.string(),
website: z.string().url().nullable().catch(null),
}).nullable().catch(null),
bookmarked: z.coerce.boolean(),
card: cardSchema.nullable().catch(null),
content: contentSchema,
created_at: dateSchema,
disliked: z.coerce.boolean(),
dislikes_count: z.number().catch(0),
edited_at: z.string().datetime().nullable().catch(null),
emojis: filteredArray(customEmojiSchema),
favourited: z.coerce.boolean(),
favourites_count: z.number().catch(0),
group: groupSchema.nullable().catch(null),
in_reply_to_account_id: z.string().nullable().catch(null),
in_reply_to_id: z.string().nullable().catch(null),
id: z.string(),
language: z.string().nullable().catch(null),
media_attachments: filteredArray(attachmentSchema),
mentions: filteredArray(mentionSchema),
muted: z.coerce.boolean(),
pinned: z.coerce.boolean(),
pleroma: statusPleromaSchema.optional().catch(undefined),
2023-05-04 10:13:39 -07:00
poll: pollSchema.nullable().catch(null),
quote: z.literal(null).catch(null),
quotes_count: z.number().catch(0),
reblog: z.literal(null).catch(null),
2023-05-04 10:13:39 -07:00
reblogged: z.coerce.boolean(),
reblogs_count: z.number().catch(0),
replies_count: z.number().catch(0),
sensitive: z.coerce.boolean(),
spoiler_text: contentSchema,
tags: filteredArray(tagSchema),
2023-05-18 15:09:15 -07:00
tombstone: z.object({
reason: z.enum(['deleted']),
}).nullable().optional().catch(undefined),
2023-05-04 10:13:39 -07:00
uri: z.string().url().catch(''),
url: z.string().url().catch(''),
visibility: z.string().catch('public'),
});
2023-05-18 15:09:15 -07:00
type BaseStatus = z.infer<typeof baseStatusSchema>;
type TransformableStatus = Omit<BaseStatus, 'reblog' | 'quote' | 'pleroma'> & {
pleroma?: Omit<z.infer<typeof statusPleromaSchema>, 'quote'>
};
2023-05-18 15:09:15 -07:00
/** Creates search index from the status. */
const buildSearchIndex = (status: TransformableStatus): string => {
const pollOptionTitles = status.poll ? status.poll.options.map(({ title }) => title) : [];
const mentionedUsernames = status.mentions.map(({ acct }) => `@${acct}`);
const fields = [
status.spoiler_text,
status.content,
...pollOptionTitles,
...mentionedUsernames,
];
const searchContent = unescapeHTML(fields.join('\n\n')) || '';
return new DOMParser().parseFromString(searchContent, 'text/html').documentElement.textContent || '';
};
2023-05-19 09:13:44 -07:00
type Translation = {
content: string
provider: string
}
2023-05-18 15:09:15 -07:00
/** Add internal fields to the status. */
const transformStatus = <T extends TransformableStatus>({ pleroma, ...status }: T) => {
2023-05-11 09:56:19 -07:00
const emojiMap = makeCustomEmojiMap(status.emojis);
const contentHtml = stripCompatibilityFeatures(emojify(status.content, emojiMap));
const spoilerHtml = emojify(escapeTextContentForBrowser(status.spoiler_text), emojiMap);
return {
...status,
approval_status: 'approval' as const,
2023-05-11 09:56:19 -07:00
contentHtml,
expectsCard: false,
event: pleroma?.event,
2023-05-19 09:13:44 -07:00
filtered: [],
hidden: false,
pleroma: pleroma ? (() => {
const { event, ...rest } = pleroma;
return rest;
})() : undefined,
search_index: buildSearchIndex(status),
2023-05-19 09:13:44 -07:00
showFiltered: false, // TODO: this should be removed from the schema and done somewhere else
spoilerHtml,
2023-05-19 09:13:44 -07:00
translation: undefined as Translation | undefined,
2023-05-11 09:56:19 -07:00
};
2023-05-18 15:09:15 -07:00
};
const embeddedStatusSchema = baseStatusSchema
.transform(transformStatus)
.nullable()
.catch(null);
const statusSchema = baseStatusSchema.extend({
quote: embeddedStatusSchema,
reblog: embeddedStatusSchema,
pleroma: statusPleromaSchema.extend({
2023-05-18 15:09:15 -07:00
quote: embeddedStatusSchema,
2023-05-19 09:13:44 -07:00
}).optional().catch(undefined),
2023-05-18 15:09:15 -07:00
}).transform(({ pleroma, ...status }) => {
return {
...status,
2023-05-19 09:13:44 -07:00
event: pleroma?.event,
quote: pleroma?.quote || status.quote || null,
// There's apparently no better way to do this...
// Just trying to remove the `event` and `quote` keys from the object.
pleroma: pleroma ? (() => {
const { event, quote, ...rest } = pleroma;
return rest;
})() : undefined,
2023-05-18 15:09:15 -07:00
};
}).transform(transformStatus);
2023-04-10 13:22:08 -07:00
2023-06-13 20:12:42 -07:00
type Status = Resolve<z.infer<typeof statusSchema>>;
2023-04-10 13:22:08 -07:00
export { statusSchema, type Status };