2022-04-25 11:57:24 -07:00
|
|
|
import { createAsyncThunk } from '@reduxjs/toolkit';
|
2022-06-16 12:32:17 -07:00
|
|
|
import get from 'lodash/get';
|
2022-10-27 10:46:03 -07:00
|
|
|
import { gte } from 'semver';
|
2022-01-10 14:25:06 -08:00
|
|
|
|
2021-10-20 13:17:02 -07:00
|
|
|
import KVStore from 'soapbox/storage/kv_store';
|
2022-04-25 11:57:24 -07:00
|
|
|
import { RootState } from 'soapbox/store';
|
2022-01-10 14:17:52 -08:00
|
|
|
import { getAuthUserUrl } from 'soapbox/utils/auth';
|
|
|
|
import { parseVersion } from 'soapbox/utils/features';
|
2022-03-18 14:04:08 -07:00
|
|
|
|
2022-01-10 14:01:24 -08:00
|
|
|
import api from '../api';
|
2021-10-20 11:18:55 -07:00
|
|
|
|
2022-03-15 06:48:18 -07:00
|
|
|
const getMeUrl = (state: RootState) => {
|
|
|
|
const me = state.me;
|
|
|
|
return state.accounts.getIn([me, 'url']);
|
2021-10-20 11:18:55 -07:00
|
|
|
};
|
|
|
|
|
2022-04-25 11:57:24 -07:00
|
|
|
/** Figure out the appropriate instance to fetch depending on the state */
|
2022-03-15 06:48:18 -07:00
|
|
|
export const getHost = (state: RootState) => {
|
2021-10-20 11:18:55 -07:00
|
|
|
const accountUrl = getMeUrl(state) || getAuthUserUrl(state);
|
|
|
|
|
|
|
|
try {
|
|
|
|
return new URL(accountUrl).host;
|
|
|
|
} catch {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-04-25 11:57:24 -07:00
|
|
|
export const rememberInstance = createAsyncThunk(
|
|
|
|
'instance/remember',
|
|
|
|
async(host: string) => {
|
|
|
|
return await KVStore.getItemOrError(`instance:${host}`);
|
|
|
|
},
|
|
|
|
);
|
2021-10-20 11:18:55 -07:00
|
|
|
|
2022-04-25 11:57:24 -07:00
|
|
|
/** We may need to fetch nodeinfo on Pleroma < 2.1 */
|
2022-03-15 06:48:18 -07:00
|
|
|
const needsNodeinfo = (instance: Record<string, any>): boolean => {
|
2021-10-20 13:17:02 -07:00
|
|
|
const v = parseVersion(get(instance, 'version'));
|
|
|
|
return v.software === 'Pleroma' && !get(instance, ['pleroma', 'metadata']);
|
|
|
|
};
|
|
|
|
|
2022-10-27 10:46:03 -07:00
|
|
|
/** Mastodon exposes features availabiliy under /api/v2/instance since 4.0.0 */
|
|
|
|
const supportsInstanceV2 = (instance: Record<string, any>): boolean => {
|
|
|
|
const v = parseVersion(get(instance, 'version'));
|
|
|
|
return v.software === 'Mastodon' && gte(v.compatVersion, '4.0.0');
|
|
|
|
};
|
|
|
|
|
2022-04-25 11:57:24 -07:00
|
|
|
export const fetchInstance = createAsyncThunk<void, void, { state: RootState }>(
|
|
|
|
'instance/fetch',
|
2022-05-02 09:46:18 -07:00
|
|
|
async(_arg, { dispatch, getState, rejectWithValue }) => {
|
|
|
|
try {
|
|
|
|
const { data: instance } = await api(getState).get('/api/v1/instance');
|
|
|
|
if (needsNodeinfo(instance)) {
|
|
|
|
dispatch(fetchNodeinfo());
|
|
|
|
}
|
2022-10-27 10:46:03 -07:00
|
|
|
if (supportsInstanceV2(instance)) {
|
|
|
|
dispatch(fetchInstanceV2());
|
|
|
|
}
|
2022-05-02 09:46:18 -07:00
|
|
|
return instance;
|
2022-05-11 10:40:34 -07:00
|
|
|
} catch (e) {
|
2022-05-02 09:46:18 -07:00
|
|
|
return rejectWithValue(e);
|
2022-04-25 11:57:24 -07:00
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
2020-04-01 13:05:52 -07:00
|
|
|
|
2022-04-25 11:57:24 -07:00
|
|
|
/** Tries to remember the instance from browser storage before fetching it */
|
|
|
|
export const loadInstance = createAsyncThunk<void, void, { state: RootState }>(
|
|
|
|
'instance/load',
|
|
|
|
async(_arg, { dispatch, getState }) => {
|
2021-10-20 11:18:55 -07:00
|
|
|
const host = getHost(getState());
|
2022-04-25 11:57:24 -07:00
|
|
|
await Promise.all([
|
|
|
|
dispatch(rememberInstance(host || '')),
|
|
|
|
dispatch(fetchInstance()),
|
|
|
|
]);
|
|
|
|
},
|
|
|
|
);
|
2021-10-20 11:18:55 -07:00
|
|
|
|
2022-10-27 10:46:03 -07:00
|
|
|
export const fetchInstanceV2 = createAsyncThunk<void, void, { state: RootState }>(
|
2022-04-25 11:57:24 -07:00
|
|
|
'nodeinfo/fetch',
|
|
|
|
async(_arg, { getState }) => {
|
2022-10-27 10:46:03 -07:00
|
|
|
const { data: instance } = await api(getState).get('/api/v2/instance');
|
|
|
|
return instance;
|
2022-04-25 11:57:24 -07:00
|
|
|
},
|
|
|
|
);
|
2022-10-27 10:46:03 -07:00
|
|
|
|
|
|
|
export const fetchNodeinfo = createAsyncThunk<void, void, { state: RootState }>(
|
|
|
|
'nodeinfo/fetch',
|
|
|
|
async(_arg, { getState }) => await api(getState).get('/nodeinfo/2.1.json'),
|
|
|
|
);
|