/** * Status normalizer: * Converts API statuses into our internal format. * @see {@link https://docs.joinmastodon.org/entities/status/} */ import { Map as ImmutableMap, List as ImmutableList, Record as ImmutableRecord, fromJS, } from 'immutable'; import { normalizeAttachment } from 'soapbox/normalizers/attachment'; import { normalizeCard } from 'soapbox/normalizers/card'; import { normalizeEmoji } from 'soapbox/normalizers/emoji'; import { normalizeMention } from 'soapbox/normalizers/mention'; import { normalizePoll } from 'soapbox/normalizers/poll'; import type { ReducerAccount } from 'soapbox/reducers/accounts'; import type { Account, Attachment, Card, Emoji, Mention, Poll, EmbeddedEntity } from 'soapbox/types/entities'; export type StatusVisibility = 'public' | 'unlisted' | 'private' | 'direct' | 'self'; // https://docs.joinmastodon.org/entities/status/ export const StatusRecord = ImmutableRecord({ account: null as EmbeddedEntity, application: null as ImmutableMap | null, bookmarked: false, card: null as Card | null, content: '', created_at: '', edited_at: null as string | null, emojis: ImmutableList(), favourited: false, favourites_count: 0, group: null as EmbeddedEntity, in_reply_to_account_id: null as string | null, in_reply_to_id: null as string | null, id: '', language: null as string | null, media_attachments: ImmutableList(), mentions: ImmutableList(), muted: false, pinned: false, pleroma: ImmutableMap(), poll: null as EmbeddedEntity, quote: null as EmbeddedEntity, quotes_count: 0, reblog: null as EmbeddedEntity, reblogged: false, reblogs_count: 0, replies_count: 0, sensitive: false, spoiler_text: '', tags: ImmutableList>(), uri: '', url: '', visibility: 'public' as StatusVisibility, // Internal fields contentHtml: '', expectsCard: false, filtered: false, hidden: false, search_index: '', spoilerHtml: '', translation: null as ImmutableMap | null, }); const normalizeAttachments = (status: ImmutableMap) => { return status.update('media_attachments', ImmutableList(), attachments => { return attachments.map(normalizeAttachment); }); }; const normalizeMentions = (status: ImmutableMap) => { return status.update('mentions', ImmutableList(), mentions => { return mentions.map(normalizeMention); }); }; // Normalize emojis const normalizeEmojis = (entity: ImmutableMap) => { return entity.update('emojis', ImmutableList(), emojis => { return emojis.map(normalizeEmoji); }); }; // Normalize the poll in the status, if applicable const normalizeStatusPoll = (status: ImmutableMap) => { if (status.hasIn(['poll', 'options'])) { return status.update('poll', ImmutableMap(), normalizePoll); } else { return status.set('poll', null); } }; // Normalize card const normalizeStatusCard = (status: ImmutableMap) => { if (status.get('card')) { return status.update('card', ImmutableMap(), normalizeCard); } else { return status.set('card', null); } }; // Fix order of mentions const fixMentionsOrder = (status: ImmutableMap) => { const mentions = status.get('mentions', ImmutableList()); const inReplyToAccountId = status.get('in_reply_to_account_id'); // Sort the replied-to mention to the top const sorted = mentions.sort((a: ImmutableMap, _b: ImmutableMap) => { if (a.get('id') === inReplyToAccountId) { return -1; } else { return 0; } }); return status.set('mentions', sorted); }; // Add self to mentions if it's a reply to self const addSelfMention = (status: ImmutableMap) => { const accountId = status.getIn(['account', 'id']); const isSelfReply = accountId === status.get('in_reply_to_account_id'); const hasSelfMention = accountId === status.getIn(['mentions', 0, 'id']); if (isSelfReply && !hasSelfMention && accountId) { const mention = normalizeMention(status.get('account')); return status.update('mentions', ImmutableList(), mentions => ( ImmutableList([mention]).concat(mentions) )); } else { return status; } }; // Move the quote to the top-level const fixQuote = (status: ImmutableMap) => { return status.withMutations(status => { status.update('quote', quote => quote || status.getIn(['pleroma', 'quote']) || null); status.deleteIn(['pleroma', 'quote']); status.update('quotes_count', quotes_count => quotes_count || status.getIn(['pleroma', 'quotes_count'], 0)); status.deleteIn(['pleroma', 'quotes_count']); }); }; // Workaround for not yet implemented filtering from Mastodon 3.6 const fixFiltered = (status: ImmutableMap) => { status.delete('filtered'); }; /** If the status contains spoiler text, treat it as sensitive. */ const fixSensitivity = (status: ImmutableMap) => { if (status.get('spoiler_text')) { status.set('sensitive', true); } }; export const normalizeStatus = (status: Record) => { return StatusRecord( ImmutableMap(fromJS(status)).withMutations(status => { normalizeAttachments(status); normalizeMentions(status); normalizeEmojis(status); normalizeStatusPoll(status); normalizeStatusCard(status); fixMentionsOrder(status); addSelfMention(status); fixQuote(status); fixFiltered(status); fixSensitivity(status); }), ); };