import { Map as ImmutableMap, List as ImmutableList, OrderedSet as ImmutableOrderedSet, fromJS, } from 'immutable'; import { createSelector } from 'reselect'; import { getSettings } from 'soapbox/actions/settings'; import { getDomain } from 'soapbox/utils/accounts'; import { validId } from 'soapbox/utils/auth'; import ConfigDB from 'soapbox/utils/config_db'; import { shouldFilter } from 'soapbox/utils/timelines'; import type { ReducerChat } from 'soapbox/reducers/chats'; import type { RootState } from 'soapbox/store'; import type { Notification } from 'soapbox/types/entities'; const normalizeId = (id: any): string => typeof id === 'string' ? id : ''; const getAccountBase = (state: RootState, id: string) => state.accounts.get(id); const getAccountCounters = (state: RootState, id: string) => state.accounts_counters.get(id); const getAccountRelationship = (state: RootState, id: string) => state.relationships.get(id); const getAccountMoved = (state: RootState, id: string) => state.accounts.get(state.accounts.get(id)?.moved || ''); const getAccountMeta = (state: RootState, id: string) => state.accounts_meta.get(id); const getAccountAdminData = (state: RootState, id: string) => state.admin.users.get(id); const getAccountPatron = (state: RootState, id: string) => { const url = state.accounts.get(id)?.url; return url ? state.patron.accounts.get(url) : null; }; export const makeGetAccount = () => { return createSelector([ getAccountBase, getAccountCounters, getAccountRelationship, getAccountMoved, getAccountMeta, getAccountAdminData, getAccountPatron, ], (base, counters, relationship, moved, meta, admin, patron) => { if (!base) return null; return base.withMutations(map => { if (counters) map.merge(counters); if (meta) { map.merge(meta); map.set('pleroma', meta.pleroma.merge(base.get('pleroma', ImmutableMap()))); // Lol, thanks Pleroma } if (relationship) map.set('relationship', relationship); map.set('moved', moved || null); map.set('patron', patron || null); map.setIn(['pleroma', 'admin'], admin); }); }); }; const findAccountsByUsername = (state: RootState, username: string) => { const accounts = state.accounts; return accounts.filter(account => { return username.toLowerCase() === account.acct.toLowerCase(); }); }; export const findAccountByUsername = (state: RootState, username: string) => { const accounts = findAccountsByUsername(state, username); if (accounts.size > 1) { const me = state.me; const meURL = state.accounts.get(me)?.url || ''; return accounts.find(account => { try { // If more than one account has the same username, try matching its host const { host } = new URL(account.url); const { host: meHost } = new URL(meURL); return host === meHost; } catch { return false; } }); } else { return accounts.first(); } }; const toServerSideType = (columnType: string): string => { switch (columnType) { case 'home': case 'notifications': case 'public': case 'thread': return columnType; default: if (columnType.indexOf('list:') > -1) { return 'home'; } else { return 'public'; // community, account, hashtag } } }; type FilterContext = { contextType?: string }; export const getFilters = (state: RootState, query: FilterContext) => { return state.filters.filter((filter): boolean => { return query?.contextType && filter.get('context').includes(toServerSideType(query.contextType)) && (filter.get('expires_at') === null || Date.parse(filter.get('expires_at')) > new Date().getTime()); }); }; const escapeRegExp = (string: string) => string.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'); // $& means the whole matched string export const regexFromFilters = (filters: ImmutableList>) => { if (filters.size === 0) return null; return new RegExp(filters.map(filter => { let expr = escapeRegExp(filter.get('phrase')); if (filter.get('whole_word')) { if (/^[\w]/.test(expr)) { expr = `\\b${expr}`; } if (/[\w]$/.test(expr)) { expr = `${expr}\\b`; } } return expr; }).join('|'), 'i'); }; type APIStatus = { id: string, username?: string }; export const makeGetStatus = () => { return createSelector( [ (state: RootState, { id }: APIStatus) => state.statuses.get(id), (state: RootState, { id }: APIStatus) => state.statuses.get(state.statuses.get(id)?.reblog || ''), (state: RootState, { id }: APIStatus) => state.accounts.get(state.statuses.get(id)?.account || ''), (state: RootState, { id }: APIStatus) => state.accounts.get(state.statuses.get(state.statuses.get(id)?.reblog || '')?.account || ''), (_state: RootState, { username }: APIStatus) => username, getFilters, (state: RootState) => state.me, ], (statusBase, statusReblog, accountBase, accountReblog, username, filters, me) => { if (!statusBase || !accountBase) return null; const accountUsername = accountBase.acct; //Must be owner of status if username exists if (accountUsername !== username && username !== undefined) { return null; } if (statusReblog && accountReblog) { // @ts-ignore AAHHHHH statusReblog = statusReblog.set('account', accountReblog); } else { statusReblog = undefined; } const regex = (accountReblog || accountBase).id !== me && regexFromFilters(filters); const filtered = regex && regex.test(statusReblog?.search_index || statusBase.search_index); return statusBase.withMutations(map => { map.set('reblog', statusReblog || null); // @ts-ignore :( map.set('account', accountBase || null); map.set('filtered', Boolean(filtered)); }); }, ); }; const getAlertsBase = (state: RootState) => state.alerts; const buildAlert = (item: any) => { return { message: item.message, title: item.title, actionLabel: item.actionLabel, actionLink: item.actionLink, key: item.key, className: `notification-bar-${item.severity}`, activeClassName: 'snackbar--active', dismissAfter: 6000, style: false, }; }; type Alert = ReturnType; export const getAlerts = createSelector([getAlertsBase], (base): Alert[] => { const arr: Alert[] = []; base.forEach(item => arr.push(buildAlert(item))); return arr; }); export const makeGetNotification = () => { return createSelector([ (_state: RootState, notification: Notification) => notification, (state: RootState, notification: Notification) => state.accounts.get(normalizeId(notification.account)), (state: RootState, notification: Notification) => state.accounts.get(normalizeId(notification.target)), (state: RootState, notification: Notification) => state.statuses.get(normalizeId(notification.status)), ], (notification, account, target, status) => { return notification.merge({ // @ts-ignore account: account || null, // @ts-ignore target: target || null, // @ts-ignore status: status || null, }); }); }; export const getAccountGallery = createSelector([ (state: RootState, id: string) => state.timelines.getIn([`account:${id}:media`, 'items'], ImmutableList()), (state: RootState) => state.statuses, (state: RootState) => state.accounts, ], (statusIds, statuses, accounts) => { return statusIds.reduce((medias: ImmutableList, statusId: string) => { const status = statuses.get(statusId); if (!status) return medias; if (status.reblog) return medias; if (typeof status.account !== 'string') return medias; const account = accounts.get(status.account); return medias.concat( status.media_attachments.map(media => media.merge({ status, account }))); }, ImmutableList()); }); type APIChat = { id: string, last_message: string }; export const makeGetChat = () => { return createSelector( [ (state: RootState, { id }: APIChat) => state.chats.getIn(['items', id]) as ReducerChat, (state: RootState, { id }: APIChat) => state.accounts.get(state.chats.getIn(['items', id, 'account'])), (state: RootState, { last_message }: APIChat) => state.chat_messages.get(last_message), ], (chat, account, lastMessage) => { if (!chat || !account) return null; return chat.withMutations((map) => { // @ts-ignore map.set('account', account); // @ts-ignore map.set('last_message', lastMessage); }); }, ); }; export const makeGetReport = () => { const getStatus = makeGetStatus(); return createSelector( [ (state: RootState, id: string) => state.admin.reports.get(id), (state: RootState, id: string) => state.accounts.get(state.admin.reports.get(id)?.account || ''), (state: RootState, id: string) => state.accounts.get(state.admin.reports.get(id)?.target_account || ''), // (state: RootState, id: string) => state.accounts.get(state.admin.reports.get(id)?.action_taken_by_account || ''), // (state: RootState, id: string) => state.accounts.get(state.admin.reports.get(id)?.assigned_account || ''), (state: RootState, id: string) => ImmutableList(fromJS(state.admin.reports.get(id)?.statuses)).map( statusId => state.statuses.get(normalizeId(statusId))) .filter((s: any) => s) .map((s: any) => getStatus(state, s.toJS())), ], (report, account, targetAccount, statuses) => { if (!report) return null; return report.withMutations((report) => { // @ts-ignore report.set('account', account); // @ts-ignore report.set('target_account', targetAccount); // @ts-ignore report.set('statuses', statuses); }); }, ); }; const getAuthUserIds = createSelector([ (state: RootState) => state.auth.get('users', ImmutableMap()), ], authUsers => { return authUsers.reduce((ids: ImmutableOrderedSet, authUser: ImmutableMap) => { try { const id = authUser.get('id'); return validId(id) ? ids.add(id) : ids; } catch { return ids; } }, ImmutableOrderedSet()); }); export const makeGetOtherAccounts = () => { return createSelector([ (state: RootState) => state.accounts, getAuthUserIds, (state: RootState) => state.me, ], (accounts, authUserIds, me) => { return authUserIds .reduce((list: ImmutableList, id: string) => { if (id === me) return list; const account = accounts.get(id); return account ? list.push(account) : list; }, ImmutableList()); }); }; const getSimplePolicy = createSelector([ (state: RootState) => state.admin.configs, (state: RootState) => state.instance.pleroma.getIn(['metadata', 'federation', 'mrf_simple'], ImmutableMap()) as ImmutableMap, ], (configs, instancePolicy: ImmutableMap) => { return instancePolicy.merge(ConfigDB.toSimplePolicy(configs)); }); const getRemoteInstanceFavicon = (state: RootState, host: string) => ( (state.accounts.find(account => getDomain(account) === host, null) || ImmutableMap()) .getIn(['pleroma', 'favicon']) ); const getRemoteInstanceFederation = (state: RootState, host: string) => ( getSimplePolicy(state) .map(hosts => hosts.includes(host)) ); export const makeGetHosts = () => { return createSelector([getSimplePolicy], (simplePolicy) => { return simplePolicy .deleteAll(['accept', 'reject_deletes', 'report_removal']) .reduce((acc, hosts) => acc.union(hosts), ImmutableOrderedSet()) .sort(); }); }; export const makeGetRemoteInstance = () => { return createSelector([ (_state: RootState, host: string) => host, getRemoteInstanceFavicon, getRemoteInstanceFederation, ], (host, favicon, federation) => { return ImmutableMap({ host, favicon, federation, }); }); }; type ColumnQuery = { type: string, prefix?: string }; export const makeGetStatusIds = () => createSelector([ (state: RootState, { type, prefix }: ColumnQuery) => getSettings(state).get(prefix || type, ImmutableMap()), (state: RootState, { type }: ColumnQuery) => state.timelines.getIn([type, 'items'], ImmutableOrderedSet()), (state: RootState) => state.statuses, ], (columnSettings, statusIds: ImmutableOrderedSet, statuses) => { return statusIds.filter((id: string) => { const status = statuses.get(id); if (!status) return true; return !shouldFilter(status, columnSettings); }); });