Merge branch 'next-ts-features' into 'next'

Next: convert utils/features.ts to TypeScript

See merge request soapbox-pub/soapbox-fe!1152
This commit is contained in:
Alex Gleason 2022-03-26 19:40:37 +00:00
commit 4fec5b3dd7
6 changed files with 65 additions and 30 deletions

View file

@ -23,6 +23,7 @@ describe('normalizeInstance()', () => {
description: '',
description_limit: 1500,
email: '',
feature_quote: false,
fedibird_capabilities: [],
invites_enabled: false,
languages: [],

View file

@ -35,6 +35,7 @@ export const InstanceRecord = ImmutableRecord({
description: '',
description_limit: 1500,
email: '',
feature_quote: false,
fedibird_capabilities: ImmutableList(),
invites_enabled: false,
languages: ImmutableList(),

View file

@ -1,4 +1,4 @@
import { Map as ImmutableMap } from 'immutable';
import { InstanceRecord } from 'soapbox/normalizers';
import {
parseVersion,
@ -28,7 +28,7 @@ describe('parseVersion', () => {
describe('getFeatures', () => {
describe('emojiReacts', () => {
it('is true for Pleroma 2.0+', () => {
const instance = ImmutableMap({
const instance = InstanceRecord({
version: '2.7.2 (compatible; Pleroma 2.0.5-6-ga36eb5ea-plerasstodon+dev)',
});
const features = getFeatures(instance);
@ -36,7 +36,7 @@ describe('getFeatures', () => {
});
it('is false for Pleroma < 2.0', () => {
const instance = ImmutableMap({
const instance = InstanceRecord({
version: '2.7.2 (compatible; Pleroma 1.1.50-42-g3d9ac6ae-develop)',
});
const features = getFeatures(instance);
@ -44,7 +44,7 @@ describe('getFeatures', () => {
});
it('is false for Mastodon', () => {
const instance = ImmutableMap({ version: '3.1.4' });
const instance = InstanceRecord({ version: '3.1.4' });
const features = getFeatures(instance);
expect(features.emojiReacts).toBe(false);
});
@ -52,19 +52,19 @@ describe('getFeatures', () => {
describe('suggestions', () => {
it('is true for Mastodon 2.4.3+', () => {
const instance = ImmutableMap({ version: '2.4.3' });
const instance = InstanceRecord({ version: '2.4.3' });
const features = getFeatures(instance);
expect(features.suggestions).toBe(true);
});
it('is false for Mastodon < 2.4.3', () => {
const instance = ImmutableMap({ version: '2.4.2' });
const instance = InstanceRecord({ version: '2.4.2' });
const features = getFeatures(instance);
expect(features.suggestions).toBe(false);
});
it('is false for Pleroma', () => {
const instance = ImmutableMap({
const instance = InstanceRecord({
version: '2.7.2 (compatible; Pleroma 1.1.50-42-g3d9ac6ae-develop)',
});
const features = getFeatures(instance);
@ -74,19 +74,19 @@ describe('getFeatures', () => {
describe('trends', () => {
it('is true for Mastodon 3.0.0+', () => {
const instance = ImmutableMap({ version: '3.0.0' });
const instance = InstanceRecord({ version: '3.0.0' });
const features = getFeatures(instance);
expect(features.trends).toBe(true);
});
it('is false for Mastodon < 3.0.0', () => {
const instance = ImmutableMap({ version: '2.4.3' });
const instance = InstanceRecord({ version: '2.4.3' });
const features = getFeatures(instance);
expect(features.trends).toBe(false);
});
it('is false for Pleroma', () => {
const instance = ImmutableMap({
const instance = InstanceRecord({
version: '2.7.2 (compatible; Pleroma 1.1.50-42-g3d9ac6ae-develop)',
});
const features = getFeatures(instance);
@ -96,13 +96,13 @@ describe('getFeatures', () => {
describe('focalPoint', () => {
it('is true for Mastodon 2.3.0+', () => {
const instance = ImmutableMap({ version: '2.3.0' });
const instance = InstanceRecord({ version: '2.3.0' });
const features = getFeatures(instance);
expect(features.focalPoint).toBe(true);
});
it('is false for Pleroma', () => {
const instance = ImmutableMap({
const instance = InstanceRecord({
version: '2.7.2 (compatible; Pleroma 1.1.50-42-g3d9ac6ae-develop)',
});
const features = getFeatures(instance);

View file

@ -6,27 +6,30 @@ import lt from 'semver/functions/lt';
import { custom } from 'soapbox/custom';
import type { Instance } from 'soapbox/types/entities';
// Import custom overrides, if exists
const overrides = custom('features');
// Truthy array convenience function
const any = arr => arr.some(Boolean);
const any = (arr: Array<any>): boolean => arr.some(Boolean);
// For uglification
export const MASTODON = 'Mastodon';
export const PLEROMA = 'Pleroma';
export const MITRA = 'Mitra';
export const MASTODON = 'Mastodon';
export const PLEROMA = 'Pleroma';
export const MITRA = 'Mitra';
export const TRUTHSOCIAL = 'TruthSocial';
export const getFeatures = createSelector([instance => instance], instance => {
const getInstanceFeatures = (instance: Instance) => {
const v = parseVersion(instance.get('version'));
const features = instance.getIn(['pleroma', 'metadata', 'features'], ImmutableList());
const federation = instance.getIn(['pleroma', 'metadata', 'federation'], ImmutableMap());
const features = instance.pleroma.getIn(['metadata', 'features'], ImmutableList()) as ImmutableList<string>;
const federation = instance.pleroma.getIn(['metadata', 'federation'], ImmutableMap()) as ImmutableMap<string, any>;
return Object.assign({
return {
media: true,
privacyScopes: true,
spoilers: true,
filters: true,
privacyScopes: v.software !== TRUTHSOCIAL,
spoilers: v.software !== TRUTHSOCIAL,
filters: v.software !== TRUTHSOCIAL,
polls: any([
v.software === MASTODON && gte(v.version, '2.8.0'),
v.software === PLEROMA,
@ -77,8 +80,14 @@ export const getFeatures = createSelector([instance => instance], instance => {
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,
securityAPI: any([
v.software === PLEROMA,
v.software === TRUTHSOCIAL,
]),
settingsStore: any([
v.software === PLEROMA,
v.software === TRUTHSOCIAL,
]),
accountAliasesAPI: v.software === PLEROMA,
resetPasswordAPI: v.software === PLEROMA,
exposableReactions: features.includes('exposable_reactions'),
@ -98,11 +107,14 @@ export const getFeatures = createSelector([instance => instance], instance => {
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'),
explicitAddressing: any([
v.software === PLEROMA && gte(v.version, '1.0.0'),
v.software === TRUTHSOCIAL,
]),
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,
instance.feature_quote === true,
]),
birthdays: v.software === PLEROMA && gte(v.version, '2.4.50'),
ethereumLogin: v.software === MITRA,
@ -111,11 +123,26 @@ export const getFeatures = createSelector([instance => instance], instance => {
v.software === MASTODON && gte(v.compatVersion, '3.2.0'),
v.software === PLEROMA && gte(v.version, '2.4.50'),
]),
}, overrides);
};
};
type Features = ReturnType<typeof getInstanceFeatures>;
export const getFeatures = createSelector([
(instance: Instance) => instance,
], (instance): Features => {
const features = getInstanceFeatures(instance);
return Object.assign(features, overrides) as Features;
});
export const parseVersion = version => {
const regex = /^([\w\.]*)(?: \(compatible; ([\w]*) (.*)\))?$/;
interface Backend {
software: string | null,
version: string,
compatVersion: string,
}
export const parseVersion = (version: string): Backend => {
const regex = /^([\w.]*)(?: \(compatible; ([\w]*) (.*)\))?$/;
const match = regex.exec(version);
if (match) {

View file

@ -74,6 +74,7 @@
"@types/react-helmet": "^6.1.5",
"@types/react-router-dom": "^5.3.3",
"@types/react-toggle": "^4.0.3",
"@types/semver": "^7.3.9",
"@types/uuid": "^8.3.4",
"array-includes": "^3.0.3",
"autoprefixer": "^10.4.2",

View file

@ -2169,6 +2169,11 @@
dependencies:
schema-utils "*"
"@types/semver@^7.3.9":
version "7.3.9"
resolved "https://registry.yarnpkg.com/@types/semver/-/semver-7.3.9.tgz#152c6c20a7688c30b967ec1841d31ace569863fc"
integrity sha512-L/TMpyURfBkf+o/526Zb6kd/tchUP3iBDEPjqjb+U2MAJhVRxxrmr2fwpe08E7QsV7YLcpq0tUaQ9O9x97ZIxQ==
"@types/stack-utils@^2.0.0":
version "2.0.1"
resolved "https://registry.yarnpkg.com/@types/stack-utils/-/stack-utils-2.0.1.tgz#20f18294f797f2209b5f65c8e3b5c8e8261d127c"