/** * Instance normalizer: * Converts API instances into our internal format. * @see {@link https://docs.joinmastodon.org/entities/instance/} */ import { Map as ImmutableMap, List as ImmutableList, Record as ImmutableRecord, fromJS, } from 'immutable'; import { parseVersion, PLEROMA } from 'soapbox/utils/features'; import { mergeDefined } from 'soapbox/utils/normalizers'; import { isNumber } from 'soapbox/utils/numbers'; // Use Mastodon defaults // https://docs.joinmastodon.org/entities/instance/ export const InstanceRecord = ImmutableRecord({ approval_required: false, contact_account: ImmutableMap(), configuration: ImmutableMap({ media_attachments: ImmutableMap(), chats: ImmutableMap({ max_characters: 500, }), polls: ImmutableMap({ max_options: 4, max_characters_per_option: 25, min_expiration: 300, max_expiration: 2629746, }), statuses: ImmutableMap({ max_characters: 500, max_media_attachments: 4, }), }), description: '', description_limit: 1500, email: '', feature_quote: false, fedibird_capabilities: ImmutableList(), invites_enabled: false, languages: ImmutableList(), login_message: '', pleroma: ImmutableMap({ metadata: ImmutableMap({ account_activation_required: false, birthday_min_age: 0, birthday_required: false, features: ImmutableList(), federation: ImmutableMap({ enabled: true, exclusions: false, }), }), stats: ImmutableMap(), }), registrations: false, rules: ImmutableList(), short_description: '', stats: ImmutableMap({ domain_count: 0, status_count: 0, user_count: 0, }), title: '', thumbnail: '', uri: '', urls: ImmutableMap(), version: '0.0.0', }); // Build Mastodon configuration from Pleroma instance const pleromaToMastodonConfig = (instance: ImmutableMap) => { return ImmutableMap({ statuses: ImmutableMap({ max_characters: instance.get('max_toot_chars'), }), polls: ImmutableMap({ max_options: instance.getIn(['poll_limits', 'max_options']), max_characters_per_option: instance.getIn(['poll_limits', 'max_option_chars']), min_expiration: instance.getIn(['poll_limits', 'min_expiration']), max_expiration: instance.getIn(['poll_limits', 'max_expiration']), }), }); }; // Get the software's default attachment limit const getAttachmentLimit = (software: string | null) => software === PLEROMA ? Infinity : 4; // Normalize version const normalizeVersion = (instance: ImmutableMap) => { return instance.update('version', '0.0.0', version => { // Handle Mastodon release candidates if (new RegExp(/[0-9\.]+rc[0-9]+/g).test(version)) { return version.split('rc').join('-rc'); } else { return version; } }); }; /** Rename Akkoma to Pleroma+akkoma */ const fixTakahe = (instance: ImmutableMap) => { const version: string = instance.get('version', ''); if (version.startsWith('takahe/')) { return instance.set('version', `0.0.0 (compatible; takahe ${version.slice(7)})`); } else { return instance; } }; /** Rename Akkoma to Pleroma+akkoma */ const fixAkkoma = (instance: ImmutableMap) => { const version: string = instance.get('version', ''); if (version.includes('Akkoma')) { return instance.set('version', '2.7.2 (compatible; Pleroma 2.4.50+akkoma)'); } else { return instance; } }; // Normalize instance (Pleroma, Mastodon, etc.) to Mastodon's format export const normalizeInstance = (instance: Record) => { return InstanceRecord( ImmutableMap(fromJS(instance)).withMutations((instance: ImmutableMap) => { const { software } = parseVersion(instance.get('version')); const mastodonConfig = pleromaToMastodonConfig(instance); // Merge configuration instance.update('configuration', ImmutableMap(), configuration => ( configuration.mergeDeepWith(mergeDefined, mastodonConfig) )); // If max attachments isn't set, check the backend software instance.updateIn(['configuration', 'statuses', 'max_media_attachments'], value => { return isNumber(value) ? value : getAttachmentLimit(software); }); // Normalize version normalizeVersion(instance); fixTakahe(instance); fixAkkoma(instance); // Merge defaults instance.mergeDeepWith(mergeDefined, InstanceRecord()); }), ); };