import { importEntities } from 'soapbox/entity-store/actions'; import { Entities } from 'soapbox/entity-store/entities'; import { Group, groupSchema } from 'soapbox/schemas'; import { filteredArray } from 'soapbox/schemas/utils'; 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 }); const importAccounts = (accounts: APIEntity[]) => ({ type: ACCOUNTS_IMPORT, accounts }); const importGroup = (group: Group) => importEntities([group], Entities.GROUPS); const importGroups = (groups: Group[]) => importEntities(groups, Entities.GROUPS); const importStatus = (status: APIEntity, idempotencyKey?: string) => (dispatch: AppDispatch, getState: () => RootState) => { const expandSpoilers = getSettings(getState()).get('expandSpoilers'); return dispatch({ type: STATUS_IMPORT, status, idempotencyKey, expandSpoilers }); }; const importStatuses = (statuses: APIEntity[]) => (dispatch: AppDispatch, getState: () => RootState) => { const expandSpoilers = getSettings(getState()).get('expandSpoilers'); return dispatch({ type: STATUSES_IMPORT, statuses, expandSpoilers }); }; const importPolls = (polls: APIEntity[]) => ({ type: POLLS_IMPORT, polls }); const importFetchedAccount = (account: APIEntity) => importFetchedAccounts([account]); const importFetchedAccounts = (accounts: APIEntity[], args = { should_refetch: false }) => { const { should_refetch } = args; const normalAccounts: APIEntity[] = []; const processAccount = (account: APIEntity) => { if (!account.id) return; if (should_refetch) { account.should_refetch = true; } normalAccounts.push(account); if (account.moved) { processAccount(account.moved); } }; 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); }; 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)); } // Pleroma quotes if (status.pleroma?.quote?.id) { dispatch(importFetchedStatus(status.pleroma.quote)); } // 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)); dispatch(importStatus(status, idempotencyKey)); }; // 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[] = []; function processStatus(status: APIEntity) { // Skip broken statuses if (isBroken(status)) return; normalStatuses.push(status); accounts.push(status.account); if (status.reblog?.id) { 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) { polls.push(status.poll); } if (status.group?.id) { dispatch(importFetchedGroup(status.group)); } } statuses.forEach(processStatus); dispatch(importPolls(polls)); dispatch(importFetchedAccounts(accounts)); dispatch(importStatuses(normalStatuses)); }; const importFetchedPoll = (poll: APIEntity) => (dispatch: AppDispatch) => { dispatch(importPolls([poll])); }; 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, };