pleroma/app/soapbox/normalizers/account.ts

296 lines
9 KiB
TypeScript
Raw Normal View History

2022-03-12 13:01:00 -08:00
/**
* Account normalizer:
* Converts API accounts into our internal format.
* @see {@link https://docs.joinmastodon.org/entities/account/}
*/
2022-03-11 18:48:00 -08:00
import escapeTextContentForBrowser from 'escape-html';
2022-03-09 14:00:43 -08:00
import {
Map as ImmutableMap,
List as ImmutableList,
Record as ImmutableRecord,
2022-03-16 19:33:09 -07:00
fromJS,
2022-03-09 14:00:43 -08:00
} from 'immutable';
2022-03-11 18:48:00 -08:00
import emojify from 'soapbox/features/emoji/emoji';
import { normalizeEmoji } from 'soapbox/normalizers/emoji';
import { unescapeHTML } from 'soapbox/utils/html';
import { mergeDefined, makeEmojiMap } from 'soapbox/utils/normalizers';
import type { PatronAccount } from 'soapbox/reducers/patron';
import type { Emoji, Field, EmbeddedEntity, Relationship } from 'soapbox/types/entities';
2022-03-11 18:48:00 -08:00
// https://docs.joinmastodon.org/entities/account/
2022-03-16 19:15:38 -07:00
export const AccountRecord = ImmutableRecord({
2022-03-08 21:25:30 -08:00
acct: '',
avatar: '',
avatar_static: '',
birthday: '',
2022-03-08 21:25:30 -08:00
bot: false,
2022-08-31 15:01:19 -07:00
created_at: '',
discoverable: false,
2022-03-08 21:25:30 -08:00
display_name: '',
emojis: ImmutableList<Emoji>(),
favicon: '',
fields: ImmutableList<Field>(),
2022-03-08 21:25:30 -08:00
followers_count: 0,
following_count: 0,
fqn: '',
header: '',
header_static: '',
id: '',
2022-08-31 15:01:19 -07:00
last_status_at: '',
2022-03-08 21:25:30 -08:00
location: '',
locked: false,
moved: null as EmbeddedEntity<any>,
2022-11-04 10:20:55 -07:00
mute_expires_at: null as string | null,
2022-03-08 21:25:30 -08:00
note: '',
pleroma: ImmutableMap<string, any>(),
source: ImmutableMap<string, any>(),
2022-03-08 21:25:30 -08:00
statuses_count: 0,
uri: '',
url: '',
username: '',
2022-03-17 12:49:42 -07:00
website: '',
2022-03-08 21:25:30 -08:00
verified: false,
// Internal fields
admin: false,
2022-03-08 21:25:30 -08:00
display_name_html: '',
domain: '',
moderator: false,
2022-03-08 21:25:30 -08:00
note_emojified: '',
note_plain: '',
patron: null as PatronAccount | null,
relationship: null as Relationship | null,
2022-03-08 21:25:30 -08:00
should_refetch: false,
staff: false,
2022-03-08 21:25:30 -08:00
});
2022-03-11 18:48:00 -08:00
// https://docs.joinmastodon.org/entities/field/
2022-03-16 19:15:38 -07:00
export const FieldRecord = ImmutableRecord({
2022-03-11 18:48:00 -08:00
name: '',
value: '',
verified_at: null as Date | null,
2022-03-11 18:48:00 -08:00
// Internal fields
name_emojified: '',
value_emojified: '',
value_plain: '',
});
// https://gitlab.com/soapbox-pub/soapbox/-/issues/549
const normalizePleromaLegacyFields = (account: ImmutableMap<string, any>) => {
return account.update('pleroma', ImmutableMap(), (pleroma: ImmutableMap<string, any>) => {
return pleroma.withMutations(pleroma => {
const legacy = ImmutableMap({
is_active: !pleroma.get('deactivated'),
is_confirmed: !pleroma.get('confirmation_pending'),
is_approved: !pleroma.get('approval_pending'),
});
pleroma.mergeWith(mergeDefined, legacy);
pleroma.deleteAll(['deactivated', 'confirmation_pending', 'approval_pending']);
});
});
};
2022-04-24 15:27:08 -07:00
/** Add avatar, if missing */
2022-03-11 18:48:00 -08:00
const normalizeAvatar = (account: ImmutableMap<string, any>) => {
const avatar = account.get('avatar');
const avatarStatic = account.get('avatar_static');
const missing = require('images/avatar-missing.png');
return account.withMutations(account => {
account.set('avatar', avatar || avatarStatic || missing);
account.set('avatar_static', avatarStatic || avatar || missing);
});
};
2022-04-24 15:27:08 -07:00
/** Add header, if missing */
const normalizeHeader = (account: ImmutableMap<string, any>) => {
const header = account.get('header');
const headerStatic = account.get('header_static');
const missing = require('images/header-missing.png');
return account.withMutations(account => {
account.set('header', header || headerStatic || missing);
account.set('header_static', headerStatic || header || missing);
});
};
2022-04-24 15:27:08 -07:00
/** Normalize custom fields */
2022-03-11 18:48:00 -08:00
const normalizeFields = (account: ImmutableMap<string, any>) => {
return account.update('fields', ImmutableList(), fields => fields.map(FieldRecord));
};
2022-04-24 15:27:08 -07:00
/** Normalize emojis */
2022-03-11 18:48:00 -08:00
const normalizeEmojis = (entity: ImmutableMap<string, any>) => {
const emojis = entity.get('emojis', ImmutableList()).map(normalizeEmoji);
return entity.set('emojis', emojis);
};
2022-04-24 15:27:08 -07:00
/** Normalize Pleroma/Fedibird birthday */
const normalizeBirthday = (account: ImmutableMap<string, any>) => {
2022-02-27 12:42:42 -08:00
const birthday = [
account.getIn(['pleroma', 'birthday']),
account.getIn(['other_settings', 'birthday']),
].find(Boolean);
return account.set('birthday', birthday);
};
2022-04-24 15:27:08 -07:00
/** Get Pleroma tags */
const getTags = (account: ImmutableMap<string, any>): ImmutableList<any> => {
const tags = account.getIn(['pleroma', 'tags']);
return ImmutableList(ImmutableList.isList(tags) ? tags : []);
};
2022-04-24 15:27:08 -07:00
/** Normalize Truth Social/Pleroma verified */
const normalizeVerified = (account: ImmutableMap<string, any>) => {
return account.update('verified', verified => {
return [
verified === true,
getTags(account).includes('verified'),
].some(Boolean);
});
};
/** Upgrade legacy donor tag to a badge. */
2022-04-24 15:27:08 -07:00
const normalizeDonor = (account: ImmutableMap<string, any>) => {
const tags = getTags(account);
const updated = tags.includes('donor') ? tags.push('badge:donor') : tags;
return account.setIn(['pleroma', 'tags'], updated);
2022-04-24 15:27:08 -07:00
};
/** Normalize Fedibird/Truth Social/Pleroma location */
const normalizeLocation = (account: ImmutableMap<string, any>) => {
2022-02-28 15:25:20 -08:00
return account.update('location', location => {
return [
location,
account.getIn(['pleroma', 'location']),
2022-02-28 15:25:20 -08:00
account.getIn(['other_settings', 'location']),
].find(Boolean);
});
};
2022-04-24 15:27:08 -07:00
/** Set username from acct, if applicable */
2022-03-10 15:57:12 -08:00
const fixUsername = (account: ImmutableMap<string, any>) => {
2022-03-11 18:48:00 -08:00
const acct = account.get('acct') || '';
const username = account.get('username') || '';
return account.set('username', username || acct.split('@')[0]);
};
2022-04-24 15:27:08 -07:00
/** Set display name from username, if applicable */
2022-03-11 18:48:00 -08:00
const fixDisplayName = (account: ImmutableMap<string, any>) => {
const displayName = account.get('display_name') || '';
return account.set('display_name', displayName.trim().length === 0 ? account.get('username') : displayName);
};
2022-04-24 15:27:08 -07:00
/** Emojification, etc */
2022-03-11 18:48:00 -08:00
const addInternalFields = (account: ImmutableMap<string, any>) => {
const emojiMap = makeEmojiMap(account.get('emojis'));
return account.withMutations((account: ImmutableMap<string, any>) => {
// Emojify account properties
account.merge({
display_name_html: emojify(escapeTextContentForBrowser(account.get('display_name')), emojiMap),
note_emojified: emojify(account.get('note', ''), emojiMap),
note_plain: unescapeHTML(account.get('note', '')),
});
// Emojify fields
account.update('fields', ImmutableList(), fields => {
return fields.map((field: ImmutableMap<string, any>) => {
return field.merge({
name_emojified: emojify(escapeTextContentForBrowser(field.get('name')), emojiMap),
value_emojified: emojify(field.get('value'), emojiMap),
value_plain: unescapeHTML(field.get('value')),
});
});
});
});
2022-03-10 15:57:12 -08:00
};
2022-03-31 15:00:31 -07:00
const getDomainFromURL = (account: ImmutableMap<string, any>): string => {
try {
const url = account.get('url');
return new URL(url).host;
} catch {
return '';
}
};
export const guessFqn = (account: ImmutableMap<string, any>): string => {
const acct = account.get('acct', '');
const [user, domain] = acct.split('@');
if (domain) {
return acct;
} else {
return [user, getDomainFromURL(account)].join('@');
}
};
2022-03-17 13:52:57 -07:00
const normalizeFqn = (account: ImmutableMap<string, any>) => {
2022-03-31 15:00:31 -07:00
const fqn = account.get('fqn') || guessFqn(account);
return account.set('fqn', fqn);
2022-03-17 13:52:57 -07:00
};
const normalizeFavicon = (account: ImmutableMap<string, any>) => {
const favicon = account.getIn(['pleroma', 'favicon']) || '';
return account.set('favicon', favicon);
};
const addDomain = (account: ImmutableMap<string, any>) => {
const domain = account.get('fqn', '').split('@')[1] || '';
return account.set('domain', domain);
};
const addStaffFields = (account: ImmutableMap<string, any>) => {
const admin = account.getIn(['pleroma', 'is_admin']) === true;
const moderator = account.getIn(['pleroma', 'is_moderator']) === true;
const staff = admin || moderator;
return account.merge({
admin,
moderator,
staff,
});
};
const normalizeDiscoverable = (account: ImmutableMap<string, any>) => {
const discoverable = Boolean(account.get('discoverable') || account.getIn(['source', 'pleroma', 'discoverable']));
return account.set('discoverable', discoverable);
};
/** Normalize undefined/null birthday to empty string. */
const fixBirthday = (account: ImmutableMap<string, any>) => {
const birthday = account.get('birthday');
return account.set('birthday', birthday || '');
};
2022-03-19 12:22:52 -07:00
export const normalizeAccount = (account: Record<string, any>) => {
2022-03-08 21:25:30 -08:00
return AccountRecord(
2022-03-16 19:33:09 -07:00
ImmutableMap(fromJS(account)).withMutations(account => {
2022-03-08 21:25:30 -08:00
normalizePleromaLegacyFields(account);
2022-03-11 18:48:00 -08:00
normalizeEmojis(account);
normalizeAvatar(account);
normalizeHeader(account);
2022-03-11 18:48:00 -08:00
normalizeFields(account);
2022-03-08 21:25:30 -08:00
normalizeVerified(account);
2022-04-24 15:27:08 -07:00
normalizeDonor(account);
2022-03-08 21:25:30 -08:00
normalizeBirthday(account);
normalizeLocation(account);
2022-03-17 13:52:57 -07:00
normalizeFqn(account);
normalizeFavicon(account);
normalizeDiscoverable(account);
addDomain(account);
addStaffFields(account);
2022-03-10 15:57:12 -08:00
fixUsername(account);
2022-03-11 18:48:00 -08:00
fixDisplayName(account);
fixBirthday(account);
2022-03-11 18:48:00 -08:00
addInternalFields(account);
2022-03-08 21:25:30 -08:00
}),
);
};