2022-01-10 14:01:24 -08:00
|
|
|
import { Map as ImmutableMap, List as ImmutableList, fromJS } from 'immutable';
|
2022-01-10 14:25:06 -08:00
|
|
|
|
2022-01-10 14:01:24 -08:00
|
|
|
import { ADMIN_CONFIG_UPDATE_REQUEST, ADMIN_CONFIG_UPDATE_SUCCESS } from 'soapbox/actions/admin';
|
2022-01-10 14:17:52 -08:00
|
|
|
import { PLEROMA_PRELOAD_IMPORT } from 'soapbox/actions/preload';
|
2022-01-10 14:01:24 -08:00
|
|
|
import KVStore from 'soapbox/storage/kv_store';
|
2022-01-10 14:17:52 -08:00
|
|
|
import { ConfigDB } from 'soapbox/utils/config_db';
|
2022-01-31 15:52:12 -08:00
|
|
|
import { parseVersion, PLEROMA } from 'soapbox/utils/features';
|
|
|
|
import { isNumber } from 'soapbox/utils/numbers';
|
2022-01-10 14:25:06 -08:00
|
|
|
|
2020-05-31 10:01:21 -07:00
|
|
|
import {
|
2021-10-20 11:18:55 -07:00
|
|
|
INSTANCE_REMEMBER_SUCCESS,
|
2020-06-03 12:24:53 -07:00
|
|
|
INSTANCE_FETCH_SUCCESS,
|
2021-09-04 12:47:11 -07:00
|
|
|
INSTANCE_FETCH_FAIL,
|
2020-06-03 12:24:53 -07:00
|
|
|
NODEINFO_FETCH_SUCCESS,
|
2020-05-31 10:01:21 -07:00
|
|
|
} from '../actions/instance';
|
2020-04-01 13:05:52 -07:00
|
|
|
|
2020-05-31 10:01:21 -07:00
|
|
|
const nodeinfoToInstance = nodeinfo => {
|
|
|
|
// Match Pleroma's develop branch
|
|
|
|
return ImmutableMap({
|
|
|
|
pleroma: ImmutableMap({
|
|
|
|
metadata: ImmutableMap({
|
|
|
|
account_activation_required: nodeinfo.getIn(['metadata', 'accountActivationRequired']),
|
|
|
|
features: nodeinfo.getIn(['metadata', 'features']),
|
|
|
|
federation: nodeinfo.getIn(['metadata', 'federation']),
|
2020-08-09 11:58:15 -07:00
|
|
|
fields_limits: ImmutableMap({
|
|
|
|
max_fields: nodeinfo.getIn(['metadata', 'fieldsLimits', 'maxFields']),
|
|
|
|
}),
|
2020-05-31 10:01:21 -07:00
|
|
|
}),
|
|
|
|
}),
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2022-01-31 12:26:42 -08:00
|
|
|
// Use Mastodon defaults
|
2020-04-15 11:20:09 -07:00
|
|
|
const initialState = ImmutableMap({
|
2021-07-19 14:15:42 -07:00
|
|
|
description_limit: 1500,
|
2022-01-31 12:26:42 -08:00
|
|
|
configuration: ImmutableMap({
|
|
|
|
statuses: ImmutableMap({
|
|
|
|
max_characters: 500,
|
2022-01-31 15:52:12 -08:00
|
|
|
max_media_attachments: 4,
|
2022-01-31 12:26:42 -08:00
|
|
|
}),
|
|
|
|
polls: ImmutableMap({
|
|
|
|
max_options: 4,
|
|
|
|
max_characters_per_option: 25,
|
|
|
|
min_expiration: 300,
|
|
|
|
max_expiration: 2629746,
|
|
|
|
}),
|
2020-04-15 11:20:09 -07:00
|
|
|
}),
|
2021-01-18 11:31:16 -08:00
|
|
|
version: '0.0.0',
|
2020-04-15 11:20:09 -07:00
|
|
|
});
|
2020-04-01 13:05:52 -07:00
|
|
|
|
2022-01-31 12:26:42 -08:00
|
|
|
// Build Mastodon configuration from Pleroma instance
|
|
|
|
const pleromaToMastodonConfig = instance => {
|
2022-02-10 17:34:23 -08:00
|
|
|
return ImmutableMap({
|
2022-01-31 12:26:42 -08:00
|
|
|
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']),
|
|
|
|
}),
|
2022-02-10 17:34:23 -08:00
|
|
|
});
|
2022-01-31 12:26:42 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
// Use new value only if old value is undefined
|
|
|
|
const mergeDefined = (oldVal, newVal) => oldVal === undefined ? newVal : oldVal;
|
|
|
|
|
2022-01-31 15:52:12 -08:00
|
|
|
// Get the software's default attachment limit
|
|
|
|
const getAttachmentLimit = software => software === PLEROMA ? Infinity : 4;
|
|
|
|
|
2022-01-31 12:26:42 -08:00
|
|
|
// Normalize instance (Pleroma, Mastodon, etc.) to Mastodon's format
|
|
|
|
const normalizeInstance = instance => {
|
2022-01-31 15:52:12 -08:00
|
|
|
const { software } = parseVersion(instance.get('version'));
|
2022-01-31 12:26:42 -08:00
|
|
|
const mastodonConfig = pleromaToMastodonConfig(instance);
|
|
|
|
|
|
|
|
return instance.withMutations(instance => {
|
|
|
|
// Merge configuration
|
|
|
|
instance.update('configuration', ImmutableMap(), configuration => (
|
|
|
|
configuration.mergeDeepWith(mergeDefined, mastodonConfig)
|
|
|
|
));
|
|
|
|
|
2022-01-31 15:52:12 -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);
|
|
|
|
});
|
|
|
|
|
2022-01-31 12:26:42 -08:00
|
|
|
// Merge defaults & cleanup
|
|
|
|
instance.mergeDeepWith(mergeDefined, initialState);
|
|
|
|
instance.deleteAll(['max_toot_chars', 'poll_limits']);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2021-10-20 11:18:55 -07:00
|
|
|
const importInstance = (state, instance) => {
|
2022-01-31 12:26:42 -08:00
|
|
|
return normalizeInstance(instance);
|
2021-10-20 11:18:55 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
const importNodeinfo = (state, nodeinfo) => {
|
|
|
|
return nodeinfoToInstance(nodeinfo).mergeDeep(state);
|
|
|
|
};
|
|
|
|
|
2020-08-24 13:53:38 -07:00
|
|
|
const preloadImport = (state, action, path) => {
|
2021-10-20 11:18:55 -07:00
|
|
|
const instance = action.data[path];
|
|
|
|
return instance ? importInstance(state, fromJS(instance)) : state;
|
2020-08-24 13:53:38 -07:00
|
|
|
};
|
|
|
|
|
2020-12-29 21:25:07 -08:00
|
|
|
const getConfigValue = (instanceConfig, key) => {
|
2020-12-29 21:29:06 -08:00
|
|
|
const v = instanceConfig
|
|
|
|
.find(value => value.getIn(['tuple', 0]) === key);
|
|
|
|
|
|
|
|
return v ? v.getIn(['tuple', 1]) : undefined;
|
2020-12-29 21:25:07 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
const importConfigs = (state, configs) => {
|
|
|
|
// FIXME: This is pretty hacked together. Need to make a cleaner map.
|
|
|
|
const config = ConfigDB.find(configs, ':pleroma', ':instance');
|
2021-07-28 15:06:21 -07:00
|
|
|
const simplePolicy = ConfigDB.toSimplePolicy(configs);
|
|
|
|
|
|
|
|
if (!config && !simplePolicy) return state;
|
2020-12-29 21:25:07 -08:00
|
|
|
|
|
|
|
return state.withMutations(state => {
|
2021-07-28 15:06:21 -07:00
|
|
|
if (config) {
|
|
|
|
const value = config.get('value', ImmutableList());
|
|
|
|
const registrationsOpen = getConfigValue(value, ':registrations_open');
|
|
|
|
const approvalRequired = getConfigValue(value, ':account_approval_required');
|
|
|
|
|
|
|
|
state.update('registrations', c => typeof registrationsOpen === 'boolean' ? registrationsOpen : c);
|
|
|
|
state.update('approval_required', c => typeof approvalRequired === 'boolean' ? approvalRequired : c);
|
|
|
|
}
|
2020-12-29 21:25:07 -08:00
|
|
|
|
2021-07-28 15:06:21 -07:00
|
|
|
if (simplePolicy) {
|
|
|
|
state.setIn(['pleroma', 'metadata', 'federation', 'mrf_simple'], simplePolicy);
|
|
|
|
}
|
2020-12-29 21:25:07 -08:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2021-09-04 12:47:11 -07:00
|
|
|
const handleAuthFetch = state => {
|
|
|
|
// Authenticated fetch is enabled, so make the instance appear censored
|
|
|
|
return ImmutableMap({
|
|
|
|
title: '██████',
|
|
|
|
description: '████████████',
|
|
|
|
}).merge(state);
|
|
|
|
};
|
|
|
|
|
2021-10-20 11:18:55 -07:00
|
|
|
const getHost = instance => {
|
|
|
|
try {
|
|
|
|
return new URL(instance.uri).host;
|
|
|
|
} catch {
|
|
|
|
try {
|
|
|
|
return new URL(`https://${instance.uri}`).host;
|
|
|
|
} catch {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const persistInstance = instance => {
|
|
|
|
const host = getHost(instance);
|
|
|
|
|
|
|
|
if (host) {
|
2021-10-20 13:17:02 -07:00
|
|
|
KVStore.setItem(`instance:${host}`, instance).catch(console.error);
|
2021-10-20 11:18:55 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-10-20 15:50:35 -07:00
|
|
|
const handleInstanceFetchFail = (state, error) => {
|
|
|
|
if (error.response && error.response.status === 401) {
|
|
|
|
return handleAuthFetch(state);
|
|
|
|
} else {
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2020-04-01 13:05:52 -07:00
|
|
|
export default function instance(state = initialState, action) {
|
|
|
|
switch(action.type) {
|
2021-09-15 10:15:36 -07:00
|
|
|
case PLEROMA_PRELOAD_IMPORT:
|
2020-08-24 13:53:38 -07:00
|
|
|
return preloadImport(state, action, '/api/v1/instance');
|
2021-10-20 11:18:55 -07:00
|
|
|
case INSTANCE_REMEMBER_SUCCESS:
|
|
|
|
return importInstance(state, fromJS(action.instance));
|
2020-06-03 12:24:53 -07:00
|
|
|
case INSTANCE_FETCH_SUCCESS:
|
2021-10-20 11:18:55 -07:00
|
|
|
persistInstance(action.instance);
|
|
|
|
return importInstance(state, fromJS(action.instance));
|
2021-09-04 12:47:11 -07:00
|
|
|
case INSTANCE_FETCH_FAIL:
|
2021-10-20 15:50:35 -07:00
|
|
|
return handleInstanceFetchFail(state, action.error);
|
2020-06-03 12:24:53 -07:00
|
|
|
case NODEINFO_FETCH_SUCCESS:
|
2021-10-20 11:18:55 -07:00
|
|
|
return importNodeinfo(state, fromJS(action.nodeinfo));
|
2020-12-29 21:29:06 -08:00
|
|
|
case ADMIN_CONFIG_UPDATE_REQUEST:
|
2020-12-29 21:25:07 -08:00
|
|
|
case ADMIN_CONFIG_UPDATE_SUCCESS:
|
|
|
|
return importConfigs(state, fromJS(action.configs));
|
2020-04-01 13:05:52 -07:00
|
|
|
default:
|
|
|
|
return state;
|
|
|
|
}
|
2021-08-03 12:22:51 -07:00
|
|
|
}
|