222 lines
7.1 KiB
TypeScript
222 lines
7.1 KiB
TypeScript
/**
|
|
* 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';
|
|
|
|
export type EventJoinMode = 'free' | 'restricted' | 'invite';
|
|
export type EventJoinState = 'pending' | 'reject' | 'accept';
|
|
|
|
export const EventRecord = ImmutableRecord({
|
|
name: '',
|
|
start_time: null as string | null,
|
|
end_time: null as string | null,
|
|
join_mode: null as EventJoinMode | null,
|
|
participants_count: 0,
|
|
location: null as ImmutableMap<string, any> | null,
|
|
join_state: null as EventJoinState | null,
|
|
banner: null as Attachment | null,
|
|
links: ImmutableList<Attachment>(),
|
|
});
|
|
|
|
// https://docs.joinmastodon.org/entities/status/
|
|
export const StatusRecord = ImmutableRecord({
|
|
account: null as EmbeddedEntity<Account | ReducerAccount>,
|
|
application: null as ImmutableMap<string, any> | null,
|
|
bookmarked: false,
|
|
card: null as Card | null,
|
|
content: '',
|
|
created_at: '',
|
|
edited_at: null as string | null,
|
|
emojis: ImmutableList<Emoji>(),
|
|
favourited: false,
|
|
favourites_count: 0,
|
|
group: null as EmbeddedEntity<any>,
|
|
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<Attachment>(),
|
|
mentions: ImmutableList<Mention>(),
|
|
muted: false,
|
|
pinned: false,
|
|
pleroma: ImmutableMap<string, any>(),
|
|
poll: null as EmbeddedEntity<Poll>,
|
|
quote: null as EmbeddedEntity<any>,
|
|
quotes_count: 0,
|
|
reblog: null as EmbeddedEntity<any>,
|
|
reblogged: false,
|
|
reblogs_count: 0,
|
|
replies_count: 0,
|
|
sensitive: false,
|
|
spoiler_text: '',
|
|
tags: ImmutableList<ImmutableMap<string, any>>(),
|
|
uri: '',
|
|
url: '',
|
|
visibility: 'public' as StatusVisibility,
|
|
event: null as ReturnType<typeof EventRecord> | null,
|
|
|
|
// Internal fields
|
|
contentHtml: '',
|
|
expectsCard: false,
|
|
filtered: false,
|
|
hidden: false,
|
|
search_index: '',
|
|
spoilerHtml: '',
|
|
translation: null as ImmutableMap<string, string> | null,
|
|
});
|
|
|
|
const normalizeAttachments = (status: ImmutableMap<string, any>) => {
|
|
return status.update('media_attachments', ImmutableList(), attachments => {
|
|
return attachments.map(normalizeAttachment);
|
|
});
|
|
};
|
|
|
|
const normalizeMentions = (status: ImmutableMap<string, any>) => {
|
|
return status.update('mentions', ImmutableList(), mentions => {
|
|
return mentions.map(normalizeMention);
|
|
});
|
|
};
|
|
|
|
// Normalize emojis
|
|
const normalizeEmojis = (entity: ImmutableMap<string, any>) => {
|
|
return entity.update('emojis', ImmutableList(), emojis => {
|
|
return emojis.map(normalizeEmoji);
|
|
});
|
|
};
|
|
|
|
// Normalize the poll in the status, if applicable
|
|
const normalizeStatusPoll = (status: ImmutableMap<string, any>) => {
|
|
if (status.hasIn(['poll', 'options'])) {
|
|
return status.update('poll', ImmutableMap(), normalizePoll);
|
|
} else {
|
|
return status.set('poll', null);
|
|
}
|
|
};
|
|
|
|
// Normalize card
|
|
const normalizeStatusCard = (status: ImmutableMap<string, any>) => {
|
|
if (status.get('card')) {
|
|
return status.update('card', ImmutableMap(), normalizeCard);
|
|
} else {
|
|
return status.set('card', null);
|
|
}
|
|
};
|
|
|
|
// Fix order of mentions
|
|
const fixMentionsOrder = (status: ImmutableMap<string, any>) => {
|
|
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<string, any>, _b: ImmutableMap<string, any>) => {
|
|
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<string, any>) => {
|
|
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<string, any>) => {
|
|
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<string, any>) => {
|
|
status.delete('filtered');
|
|
};
|
|
|
|
/** If the status contains spoiler text, treat it as sensitive. */
|
|
const fixSensitivity = (status: ImmutableMap<string, any>) => {
|
|
if (status.get('spoiler_text')) {
|
|
status.set('sensitive', true);
|
|
}
|
|
};
|
|
|
|
// Normalize event
|
|
const normalizeEvent = (status: ImmutableMap<string, any>) => {
|
|
if (status.getIn(['pleroma', 'event'])) {
|
|
const firstAttachment = status.get('media_attachments').first();
|
|
let banner = null;
|
|
let mediaAttachments = status.get('media_attachments');
|
|
|
|
if (firstAttachment && firstAttachment.description === 'Banner' && firstAttachment.type === 'image') {
|
|
banner = normalizeAttachment(firstAttachment);
|
|
mediaAttachments = mediaAttachments.shift();
|
|
}
|
|
|
|
const links = mediaAttachments.filter((attachment: Attachment) => attachment.pleroma.get('mime_type') === 'text/html');
|
|
mediaAttachments = mediaAttachments.filter((attachment: Attachment) => attachment.pleroma.get('mime_type') !== 'text/html');
|
|
|
|
const event = EventRecord(
|
|
(status.getIn(['pleroma', 'event']) as ImmutableMap<string, any>)
|
|
.set('banner', banner)
|
|
.set('links', links),
|
|
);
|
|
|
|
status
|
|
.set('event', event)
|
|
.set('media_attachments', mediaAttachments);
|
|
}
|
|
};
|
|
|
|
export const normalizeStatus = (status: Record<string, any>) => {
|
|
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);
|
|
normalizeEvent(status);
|
|
}),
|
|
);
|
|
};
|