131 lines
5 KiB
JavaScript
131 lines
5 KiB
JavaScript
// Detect backend features to conditionally render elements
|
|
import { List as ImmutableList, Map as ImmutableMap } from 'immutable';
|
|
import { createSelector } from 'reselect';
|
|
import gte from 'semver/functions/gte';
|
|
import lt from 'semver/functions/lt';
|
|
|
|
// FIXME: We have to use a dynamic import to trick Webpack into treating it as
|
|
// optional, but this causes custom locales to become part of the main chunk.
|
|
const importCustom = path => {
|
|
try {
|
|
return require(`custom/${path}.json`);
|
|
} catch(e) {
|
|
return {};
|
|
}
|
|
};
|
|
|
|
// Import custom overrides, if exists
|
|
const overrides = importCustom('features');
|
|
|
|
const any = arr => arr.some(Boolean);
|
|
|
|
// For uglification
|
|
export const MASTODON = 'Mastodon';
|
|
export const PLEROMA = 'Pleroma';
|
|
export const MITRA = 'Mitra';
|
|
|
|
export const getFeatures = createSelector([instance => instance], instance => {
|
|
const v = parseVersion(instance.get('version'));
|
|
const features = instance.getIn(['pleroma', 'metadata', 'features'], ImmutableList());
|
|
const federation = instance.getIn(['pleroma', 'metadata', 'federation'], ImmutableMap());
|
|
|
|
return Object.assign({
|
|
bookmarks: any([
|
|
v.software === MASTODON && gte(v.compatVersion, '3.1.0'),
|
|
v.software === PLEROMA && gte(v.version, '0.9.9'),
|
|
]),
|
|
lists: any([
|
|
v.software === MASTODON && gte(v.compatVersion, '2.1.0'),
|
|
v.software === PLEROMA && gte(v.version, '0.9.9'),
|
|
]),
|
|
suggestions: any([
|
|
v.software === MASTODON && gte(v.compatVersion, '2.4.3'),
|
|
features.includes('v2_suggestions'),
|
|
]),
|
|
suggestionsV2: any([
|
|
v.software === MASTODON && gte(v.compatVersion, '3.4.0'),
|
|
features.includes('v2_suggestions'),
|
|
]),
|
|
blockersVisible: features.includes('blockers_visible'),
|
|
trends: v.software === MASTODON && gte(v.compatVersion, '3.0.0'),
|
|
mediaV2: any([
|
|
v.software === MASTODON && gte(v.compatVersion, '3.1.3'),
|
|
// Even though Pleroma supports these endpoints, it has disadvantages
|
|
// v.software === PLEROMA && gte(v.version, '2.1.0'),
|
|
]),
|
|
directTimeline: any([
|
|
v.software === MASTODON && lt(v.compatVersion, '3.0.0'),
|
|
v.software === PLEROMA && gte(v.version, '0.9.9'),
|
|
]),
|
|
conversations: any([
|
|
v.software === MASTODON && gte(v.compatVersion, '2.6.0'),
|
|
v.software === PLEROMA && gte(v.version, '0.9.9'),
|
|
]),
|
|
emojiReacts: v.software === PLEROMA && gte(v.version, '2.0.0'),
|
|
emojiReactsRGI: v.software === PLEROMA && gte(v.version, '2.2.49'),
|
|
focalPoint: v.software === MASTODON && gte(v.compatVersion, '2.3.0'),
|
|
importAPI: v.software === PLEROMA,
|
|
importMutes: v.software === PLEROMA && gte(v.version, '2.2.0'),
|
|
emailList: features.includes('email_list'),
|
|
chats: v.software === PLEROMA && gte(v.version, '2.1.0'),
|
|
chatsV2: v.software === PLEROMA && gte(v.version, '2.3.0'),
|
|
scopes: v.software === PLEROMA ? 'read write follow push admin' : 'read write follow push',
|
|
federating: federation.get('enabled', true), // Assume true unless explicitly false
|
|
richText: v.software === PLEROMA,
|
|
securityAPI: v.software === PLEROMA,
|
|
settingsStore: v.software === PLEROMA,
|
|
accountAliasesAPI: v.software === PLEROMA,
|
|
resetPasswordAPI: v.software === PLEROMA,
|
|
exposableReactions: features.includes('exposable_reactions'),
|
|
accountSubscriptions: v.software === PLEROMA && gte(v.version, '1.0.0'),
|
|
accountNotifies: any([
|
|
v.software === MASTODON && gte(v.compatVersion, '3.3.0'),
|
|
v.software === PLEROMA && gte(v.version, '2.4.50'),
|
|
]),
|
|
unrestrictedLists: v.software === PLEROMA,
|
|
accountByUsername: v.software === PLEROMA,
|
|
profileDirectory: any([
|
|
v.software === MASTODON && gte(v.compatVersion, '3.0.0'),
|
|
features.includes('profile_directory'),
|
|
]),
|
|
accountLookup: any([
|
|
v.software === MASTODON && gte(v.compatVersion, '3.4.0'),
|
|
v.software === PLEROMA && gte(v.version, '2.4.50'),
|
|
]),
|
|
remoteInteractionsAPI: v.software === PLEROMA && gte(v.version, '2.4.50'),
|
|
explicitAddressing: v.software === PLEROMA && gte(v.version, '1.0.0'),
|
|
accountEndorsements: v.software === PLEROMA && gte(v.version, '2.4.50'),
|
|
quotePosts: any([
|
|
v.software === PLEROMA && gte(v.version, '2.4.50'),
|
|
instance.get('feature_quote') === true,
|
|
]),
|
|
birthdays: v.software === PLEROMA && gte(v.version, '2.4.50'),
|
|
ethereumLogin: v.software === MITRA,
|
|
accountMoving: v.software === PLEROMA && gte(v.version, '2.4.50'),
|
|
notes: any([
|
|
v.software === MASTODON && gte(v.compatVersion, '3.2.0'),
|
|
v.software === PLEROMA && gte(v.version, '2.4.50'),
|
|
]),
|
|
}, overrides);
|
|
});
|
|
|
|
export const parseVersion = version => {
|
|
const regex = /^([\w\.]*)(?: \(compatible; ([\w]*) (.*)\))?$/;
|
|
const match = regex.exec(version);
|
|
|
|
if (match) {
|
|
return {
|
|
software: match[2] || MASTODON,
|
|
version: match[3] || match[1],
|
|
compatVersion: match[1],
|
|
};
|
|
} else {
|
|
// If we can't parse the version, this is a new and exotic backend.
|
|
// Fall back to minimal featureset.
|
|
return {
|
|
software: null,
|
|
version: '0.0.0',
|
|
compatVersion: '0.0.0',
|
|
};
|
|
}
|
|
};
|