bigbuffet-rw/app/soapbox/actions/importer/index.ts

213 lines
5.8 KiB
TypeScript
Raw Normal View History

import { importEntities } from 'soapbox/entity-store/actions';
2023-03-16 11:15:00 -07:00
import { Entities } from 'soapbox/entity-store/entities';
import { Group, groupSchema } from 'soapbox/schemas';
import { filteredArray } from 'soapbox/schemas/utils';
2022-02-20 09:44:10 -08:00
import { getSettings } from '../settings';
import type { AppDispatch, RootState } from 'soapbox/store';
import type { APIEntity } from 'soapbox/types/entities';
const ACCOUNT_IMPORT = 'ACCOUNT_IMPORT';
const ACCOUNTS_IMPORT = 'ACCOUNTS_IMPORT';
const GROUP_IMPORT = 'GROUP_IMPORT';
const GROUPS_IMPORT = 'GROUPS_IMPORT';
const STATUS_IMPORT = 'STATUS_IMPORT';
const STATUSES_IMPORT = 'STATUSES_IMPORT';
const POLLS_IMPORT = 'POLLS_IMPORT';
const ACCOUNT_FETCH_FAIL_FOR_USERNAME_LOOKUP = 'ACCOUNT_FETCH_FAIL_FOR_USERNAME_LOOKUP';
const importAccount = (account: APIEntity) =>
({ type: ACCOUNT_IMPORT, account });
2020-03-27 13:59:38 -07:00
const importAccounts = (accounts: APIEntity[]) =>
({ type: ACCOUNTS_IMPORT, accounts });
2020-03-27 13:59:38 -07:00
const importGroup = (group: Group) =>
2023-03-16 11:15:00 -07:00
importEntities([group], Entities.GROUPS);
const importGroups = (groups: Group[]) =>
2023-03-16 11:15:00 -07:00
importEntities(groups, Entities.GROUPS);
const importStatus = (status: APIEntity, idempotencyKey?: string) =>
(dispatch: AppDispatch, getState: () => RootState) => {
2022-02-20 09:44:10 -08:00
const expandSpoilers = getSettings(getState()).get('expandSpoilers');
return dispatch({ type: STATUS_IMPORT, status, idempotencyKey, expandSpoilers });
};
2020-03-27 13:59:38 -07:00
const importStatuses = (statuses: APIEntity[]) =>
(dispatch: AppDispatch, getState: () => RootState) => {
2022-02-20 09:44:10 -08:00
const expandSpoilers = getSettings(getState()).get('expandSpoilers');
return dispatch({ type: STATUSES_IMPORT, statuses, expandSpoilers });
};
2020-03-27 13:59:38 -07:00
const importPolls = (polls: APIEntity[]) =>
({ type: POLLS_IMPORT, polls });
2020-03-27 13:59:38 -07:00
const importFetchedAccount = (account: APIEntity) =>
importFetchedAccounts([account]);
2020-03-27 13:59:38 -07:00
const importFetchedAccounts = (accounts: APIEntity[], args = { should_refetch: false }) => {
const { should_refetch } = args;
const normalAccounts: APIEntity[] = [];
2020-03-27 13:59:38 -07:00
const processAccount = (account: APIEntity) => {
if (!account.id) return;
2022-07-18 07:26:51 -07:00
if (should_refetch) {
account.should_refetch = true;
}
normalAccounts.push(account);
2020-03-27 13:59:38 -07:00
if (account.moved) {
processAccount(account.moved);
}
};
2020-03-27 13:59:38 -07:00
accounts.forEach(processAccount);
return importAccounts(normalAccounts);
};
const importFetchedGroup = (group: APIEntity) =>
importFetchedGroups([group]);
const importFetchedGroups = (groups: APIEntity[]) => {
const entities = filteredArray(groupSchema).catch([]).parse(groups);
return importGroups(entities);
};
2020-03-27 13:59:38 -07:00
const importFetchedStatus = (status: APIEntity, idempotencyKey?: string) =>
(dispatch: AppDispatch) => {
// Skip broken statuses
if (isBroken(status)) return;
if (status.reblog?.id) {
dispatch(importFetchedStatus(status.reblog));
}
// Fedibird quotes
if (status.quote?.id) {
dispatch(importFetchedStatus(status.quote));
}
2022-06-09 12:29:01 -07:00
// Pleroma quotes
if (status.pleroma?.quote?.id) {
dispatch(importFetchedStatus(status.pleroma.quote));
}
2022-06-09 12:29:01 -07:00
// Fedibird quote from reblog
if (status.reblog?.quote?.id) {
dispatch(importFetchedStatus(status.reblog.quote));
}
// Pleroma quote from reblog
if (status.reblog?.pleroma?.quote?.id) {
dispatch(importFetchedStatus(status.reblog.pleroma.quote));
}
if (status.poll?.id) {
dispatch(importFetchedPoll(status.poll));
}
if (status.group?.id) {
dispatch(importFetchedGroup(status.group));
}
dispatch(importFetchedAccount(status.account));
2022-02-19 23:27:29 -08:00
dispatch(importStatus(status, idempotencyKey));
};
2020-03-27 13:59:38 -07:00
// Sometimes Pleroma can return an empty account,
// or a repost can appear of a deleted account. Skip these statuses.
const isBroken = (status: APIEntity) => {
try {
// Skip empty accounts
// https://gitlab.com/soapbox-pub/soapbox/-/issues/424
if (!status.account.id) return true;
// Skip broken reposts
// https://gitlab.com/soapbox-pub/rebased/-/issues/28
if (status.reblog && !status.reblog.account.id) return true;
return false;
} catch (e) {
return true;
}
};
const importFetchedStatuses = (statuses: APIEntity[]) =>
(dispatch: AppDispatch, getState: () => RootState) => {
const accounts: APIEntity[] = [];
const normalStatuses: APIEntity[] = [];
const polls: APIEntity[] = [];
2020-03-27 13:59:38 -07:00
function processStatus(status: APIEntity) {
// Skip broken statuses
if (isBroken(status)) return;
2022-02-20 09:16:53 -08:00
normalStatuses.push(status);
accounts.push(status.account);
2020-03-27 13:59:38 -07:00
if (status.reblog?.id) {
2020-03-27 13:59:38 -07:00
processStatus(status.reblog);
}
// Fedibird quotes
if (status.quote?.id) {
processStatus(status.quote);
}
if (status.pleroma?.quote?.id) {
processStatus(status.pleroma.quote);
}
if (status.poll?.id) {
2022-03-10 17:55:14 -08:00
polls.push(status.poll);
2020-03-27 13:59:38 -07:00
}
if (status.group?.id) {
dispatch(importFetchedGroup(status.group));
}
2020-03-27 13:59:38 -07:00
}
statuses.forEach(processStatus);
dispatch(importPolls(polls));
dispatch(importFetchedAccounts(accounts));
2022-02-20 09:16:53 -08:00
dispatch(importStatuses(normalStatuses));
2020-03-27 13:59:38 -07:00
};
const importFetchedPoll = (poll: APIEntity) =>
(dispatch: AppDispatch) => {
2022-03-10 17:55:14 -08:00
dispatch(importPolls([poll]));
2020-03-27 13:59:38 -07:00
};
const importErrorWhileFetchingAccountByUsername = (username: string) =>
({ type: ACCOUNT_FETCH_FAIL_FOR_USERNAME_LOOKUP, username });
export {
ACCOUNT_IMPORT,
ACCOUNTS_IMPORT,
GROUP_IMPORT,
GROUPS_IMPORT,
STATUS_IMPORT,
STATUSES_IMPORT,
POLLS_IMPORT,
ACCOUNT_FETCH_FAIL_FOR_USERNAME_LOOKUP,
importAccount,
importAccounts,
importGroup,
importGroups,
importStatus,
importStatuses,
importPolls,
importFetchedAccount,
importFetchedAccounts,
importFetchedGroup,
importFetchedGroups,
importFetchedStatus,
importFetchedStatuses,
importFetchedPoll,
importErrorWhileFetchingAccountByUsername,
};