bigbuffet-rw/app/soapbox/normalizers/instance.ts

165 lines
5 KiB
TypeScript
Raw Normal View History

2022-03-12 13:01:00 -08:00
/**
* Instance normalizer:
* Converts API instances into our internal format.
* @see {@link https://docs.joinmastodon.org/entities/instance/}
*/
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';
import { parseVersion, PLEROMA } from 'soapbox/utils/features';
import { mergeDefined } from 'soapbox/utils/normalizers';
import { isNumber } from 'soapbox/utils/numbers';
// Use Mastodon defaults
2022-03-12 13:01:00 -08:00
// https://docs.joinmastodon.org/entities/instance/
2022-03-16 19:15:38 -07:00
export const InstanceRecord = ImmutableRecord({
2022-03-08 21:25:30 -08:00
approval_required: false,
2022-04-26 11:36:08 -07:00
contact_account: ImmutableMap<string, any>(),
configuration: ImmutableMap<string, any>({
media_attachments: ImmutableMap<string, any>(),
chats: ImmutableMap<string, number>({
2023-02-08 10:24:41 -08:00
max_characters: 5000,
max_media_attachments: 1,
}),
2022-04-26 11:36:08 -07:00
polls: ImmutableMap<string, number>({
max_options: 4,
max_characters_per_option: 25,
min_expiration: 300,
max_expiration: 2629746,
}),
2022-04-26 11:36:08 -07:00
statuses: ImmutableMap<string, number>({
2022-03-08 21:25:30 -08:00
max_characters: 500,
max_media_attachments: 4,
}),
groups: ImmutableMap<string, number>({
max_characters_name: 50,
2023-04-05 07:11:11 -07:00
max_characters_description: 160,
}),
2022-03-08 21:25:30 -08:00
}),
description: '',
description_limit: 1500,
email: '',
feature_quote: false,
2022-03-08 21:25:30 -08:00
fedibird_capabilities: ImmutableList(),
invites_enabled: false,
languages: ImmutableList(),
login_message: '',
2022-04-26 11:36:08 -07:00
pleroma: ImmutableMap<string, any>({
metadata: ImmutableMap<string, any>({
2022-03-08 21:25:30 -08:00
account_activation_required: false,
birthday_min_age: 0,
birthday_required: false,
features: ImmutableList(),
2022-04-26 11:36:08 -07:00
federation: ImmutableMap<string, any>({
2022-03-08 21:25:30 -08:00
enabled: true,
exclusions: false,
}),
}),
stats: ImmutableMap(),
}),
2022-03-08 21:25:30 -08:00
registrations: false,
rules: ImmutableList(),
short_description: '',
2022-04-26 11:36:08 -07:00
stats: ImmutableMap<string, number>({
domain_count: 0,
status_count: 0,
user_count: 0,
}),
2023-08-26 21:24:18 -07:00
nostr: ImmutableMap<string, any>({
relay: undefined as string | undefined,
pubkey: undefined as string | undefined,
}),
2022-03-08 21:25:30 -08:00
title: '',
thumbnail: '',
uri: '',
2022-04-26 11:36:08 -07:00
urls: ImmutableMap<string, string>(),
version: '0.0.0',
});
// Build Mastodon configuration from Pleroma instance
const pleromaToMastodonConfig = (instance: ImmutableMap<string, any>) => {
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;
2022-03-16 20:41:25 -07:00
// Normalize version
const normalizeVersion = (instance: ImmutableMap<string, any>) => {
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 fixAkkoma = (instance: ImmutableMap<string, any>) => {
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;
}
};
/** Set Takahē version to a Pleroma-like string */
const fixTakahe = (instance: ImmutableMap<string, any>) => {
2022-11-08 20:34:17 -08:00
const version: string = instance.get('version', '');
if (version.startsWith('takahe/')) {
return instance.set('version', `0.0.0 (compatible; Takahe ${version.slice(7)})`);
2022-11-08 20:34:17 -08:00
} else {
return instance;
}
};
// Normalize instance (Pleroma, Mastodon, etc.) to Mastodon's format
2022-03-16 19:33:09 -07:00
export const normalizeInstance = (instance: Record<string, any>) => {
2022-03-08 21:25:30 -08:00
return InstanceRecord(
2022-03-16 19:33:09 -07:00
ImmutableMap(fromJS(instance)).withMutations((instance: ImmutableMap<string, any>) => {
const { software } = parseVersion(instance.get('version'));
const mastodonConfig = pleromaToMastodonConfig(instance);
2022-03-08 21:25:30 -08:00
// Merge configuration
instance.update('configuration', ImmutableMap(), configuration => (
configuration.mergeDeepWith(mergeDefined, mastodonConfig)
));
2022-03-08 21:25:30 -08:00
// If max attachments isn't set, check the backend software
instance.updateIn(['configuration', 'statuses', 'max_media_attachments'], value => {
return isNumber(value) ? value : getAttachmentLimit(software);
});
// Urls can't be null, fix for Friendica
if (instance.get('urls') === null) instance.delete('urls');
2022-03-16 20:41:25 -07:00
// Normalize version
normalizeVersion(instance);
fixTakahe(instance);
2022-11-08 20:34:17 -08:00
fixAkkoma(instance);
2022-03-16 20:41:25 -07:00
// Merge defaults
2022-03-08 21:25:30 -08:00
instance.mergeDeepWith(mergeDefined, InstanceRecord());
}),
);
};