2022-07-01 13:09:07 -07:00
|
|
|
import { AxiosResponse } from 'axios';
|
|
|
|
|
2022-06-15 13:11:36 -07:00
|
|
|
import { isLoggedIn } from 'soapbox/utils/auth';
|
|
|
|
import { getFeatures } from 'soapbox/utils/features';
|
|
|
|
|
|
|
|
import api, { getLinks } from '../api';
|
|
|
|
|
|
|
|
import { fetchRelationships } from './accounts';
|
|
|
|
import { importFetchedAccounts } from './importer';
|
2022-07-01 13:09:07 -07:00
|
|
|
import { insertSuggestionsIntoTimeline } from './timelines';
|
2022-06-15 13:11:36 -07:00
|
|
|
|
|
|
|
import type { AppDispatch, RootState } from 'soapbox/store';
|
|
|
|
import type { APIEntity } from 'soapbox/types/entities';
|
|
|
|
|
|
|
|
const SUGGESTIONS_FETCH_REQUEST = 'SUGGESTIONS_FETCH_REQUEST';
|
|
|
|
const SUGGESTIONS_FETCH_SUCCESS = 'SUGGESTIONS_FETCH_SUCCESS';
|
|
|
|
const SUGGESTIONS_FETCH_FAIL = 'SUGGESTIONS_FETCH_FAIL';
|
|
|
|
|
|
|
|
const SUGGESTIONS_DISMISS = 'SUGGESTIONS_DISMISS';
|
|
|
|
|
|
|
|
const SUGGESTIONS_V2_FETCH_REQUEST = 'SUGGESTIONS_V2_FETCH_REQUEST';
|
|
|
|
const SUGGESTIONS_V2_FETCH_SUCCESS = 'SUGGESTIONS_V2_FETCH_SUCCESS';
|
|
|
|
const SUGGESTIONS_V2_FETCH_FAIL = 'SUGGESTIONS_V2_FETCH_FAIL';
|
|
|
|
|
2022-07-01 13:09:07 -07:00
|
|
|
const SUGGESTIONS_TRUTH_FETCH_REQUEST = 'SUGGESTIONS_TRUTH_FETCH_REQUEST';
|
|
|
|
const SUGGESTIONS_TRUTH_FETCH_SUCCESS = 'SUGGESTIONS_TRUTH_FETCH_SUCCESS';
|
|
|
|
const SUGGESTIONS_TRUTH_FETCH_FAIL = 'SUGGESTIONS_TRUTH_FETCH_FAIL';
|
|
|
|
|
2022-06-15 13:11:36 -07:00
|
|
|
const fetchSuggestionsV1 = (params: Record<string, any> = {}) =>
|
|
|
|
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
|
|
dispatch({ type: SUGGESTIONS_FETCH_REQUEST, skipLoading: true });
|
|
|
|
return api(getState).get('/api/v1/suggestions', { params }).then(({ data: accounts }) => {
|
|
|
|
dispatch(importFetchedAccounts(accounts));
|
|
|
|
dispatch({ type: SUGGESTIONS_FETCH_SUCCESS, accounts, skipLoading: true });
|
|
|
|
return accounts;
|
|
|
|
}).catch(error => {
|
|
|
|
dispatch({ type: SUGGESTIONS_FETCH_FAIL, error, skipLoading: true, skipAlert: true });
|
|
|
|
throw error;
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
const fetchSuggestionsV2 = (params: Record<string, any> = {}) =>
|
|
|
|
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
|
|
const next = getState().suggestions.next;
|
|
|
|
|
|
|
|
dispatch({ type: SUGGESTIONS_V2_FETCH_REQUEST, skipLoading: true });
|
|
|
|
|
|
|
|
return api(getState).get(next ? next : '/api/v2/suggestions', next ? {} : { params }).then((response) => {
|
|
|
|
const suggestions: APIEntity[] = response.data;
|
|
|
|
const accounts = suggestions.map(({ account }) => account);
|
|
|
|
const next = getLinks(response).refs.find(link => link.rel === 'next')?.uri;
|
|
|
|
|
|
|
|
dispatch(importFetchedAccounts(accounts));
|
|
|
|
dispatch({ type: SUGGESTIONS_V2_FETCH_SUCCESS, suggestions, next, skipLoading: true });
|
|
|
|
return suggestions;
|
|
|
|
}).catch(error => {
|
|
|
|
dispatch({ type: SUGGESTIONS_V2_FETCH_FAIL, error, skipLoading: true, skipAlert: true });
|
|
|
|
throw error;
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2022-07-01 13:09:07 -07:00
|
|
|
export type SuggestedProfile = {
|
|
|
|
account_avatar: string
|
|
|
|
account_id: string
|
|
|
|
acct: string
|
|
|
|
display_name: string
|
|
|
|
note: string
|
|
|
|
verified: boolean
|
|
|
|
}
|
|
|
|
|
|
|
|
const mapSuggestedProfileToAccount = (suggestedProfile: SuggestedProfile) => ({
|
|
|
|
id: suggestedProfile.account_id,
|
|
|
|
avatar: suggestedProfile.account_avatar,
|
|
|
|
avatar_static: suggestedProfile.account_avatar,
|
|
|
|
acct: suggestedProfile.acct,
|
|
|
|
display_name: suggestedProfile.display_name,
|
|
|
|
note: suggestedProfile.note,
|
|
|
|
verified: suggestedProfile.verified,
|
|
|
|
});
|
|
|
|
|
|
|
|
const fetchTruthSuggestions = (params: Record<string, any> = {}) =>
|
|
|
|
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
|
|
const next = getState().suggestions.next;
|
|
|
|
|
|
|
|
dispatch({ type: SUGGESTIONS_V2_FETCH_REQUEST, skipLoading: true });
|
|
|
|
|
|
|
|
return api(getState)
|
|
|
|
.get(next ? next : '/api/v1/truth/carousels/suggestions', next ? {} : { params })
|
|
|
|
.then((response: AxiosResponse<SuggestedProfile[]>) => {
|
|
|
|
const suggestedProfiles = response.data;
|
|
|
|
const next = getLinks(response).refs.find(link => link.rel === 'next')?.uri;
|
|
|
|
|
|
|
|
const accounts = suggestedProfiles.map(mapSuggestedProfileToAccount);
|
2022-07-18 06:54:56 -07:00
|
|
|
dispatch(importFetchedAccounts(accounts, { should_refetch: true }));
|
2022-07-01 13:09:07 -07:00
|
|
|
dispatch({ type: SUGGESTIONS_TRUTH_FETCH_SUCCESS, suggestions: suggestedProfiles, next, skipLoading: true });
|
|
|
|
return suggestedProfiles;
|
|
|
|
})
|
|
|
|
.catch(error => {
|
|
|
|
dispatch({ type: SUGGESTIONS_V2_FETCH_FAIL, error, skipLoading: true, skipAlert: true });
|
|
|
|
throw error;
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2022-06-15 13:11:36 -07:00
|
|
|
const fetchSuggestions = (params: Record<string, any> = { limit: 50 }) =>
|
|
|
|
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
|
|
const state = getState();
|
|
|
|
const me = state.me;
|
|
|
|
const instance = state.instance;
|
|
|
|
const features = getFeatures(instance);
|
|
|
|
|
2022-07-01 13:09:07 -07:00
|
|
|
if (!me) return null;
|
2022-06-15 13:11:36 -07:00
|
|
|
|
2022-07-01 13:09:07 -07:00
|
|
|
if (features.truthSuggestions) {
|
|
|
|
return dispatch(fetchTruthSuggestions(params))
|
|
|
|
.then((suggestions: APIEntity[]) => {
|
|
|
|
const accountIds = suggestions.map((account) => account.account_id);
|
|
|
|
dispatch(fetchRelationships(accountIds));
|
|
|
|
})
|
|
|
|
.catch(() => { });
|
|
|
|
} else if (features.suggestionsV2) {
|
|
|
|
return dispatch(fetchSuggestionsV2(params))
|
2022-06-15 13:11:36 -07:00
|
|
|
.then((suggestions: APIEntity[]) => {
|
|
|
|
const accountIds = suggestions.map(({ account }) => account.id);
|
|
|
|
dispatch(fetchRelationships(accountIds));
|
|
|
|
})
|
|
|
|
.catch(() => { });
|
|
|
|
} else if (features.suggestions) {
|
2022-07-01 13:09:07 -07:00
|
|
|
return dispatch(fetchSuggestionsV1(params))
|
2022-06-15 13:11:36 -07:00
|
|
|
.then((accounts: APIEntity[]) => {
|
|
|
|
const accountIds = accounts.map(({ id }) => id);
|
|
|
|
dispatch(fetchRelationships(accountIds));
|
|
|
|
})
|
|
|
|
.catch(() => { });
|
|
|
|
} else {
|
|
|
|
// Do nothing
|
2022-07-01 13:09:07 -07:00
|
|
|
return null;
|
2022-06-15 13:11:36 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-07-01 13:09:07 -07:00
|
|
|
const fetchSuggestionsForTimeline = () => (dispatch: AppDispatch, _getState: () => RootState) => {
|
|
|
|
dispatch(fetchSuggestions({ limit: 20 }))?.then(() => dispatch(insertSuggestionsIntoTimeline()));
|
|
|
|
};
|
|
|
|
|
2022-06-15 13:11:36 -07:00
|
|
|
const dismissSuggestion = (accountId: string) =>
|
|
|
|
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
|
|
if (!isLoggedIn(getState)) return;
|
|
|
|
|
|
|
|
dispatch({
|
|
|
|
type: SUGGESTIONS_DISMISS,
|
|
|
|
id: accountId,
|
|
|
|
});
|
|
|
|
|
|
|
|
api(getState).delete(`/api/v1/suggestions/${accountId}`);
|
|
|
|
};
|
|
|
|
|
|
|
|
export {
|
|
|
|
SUGGESTIONS_FETCH_REQUEST,
|
|
|
|
SUGGESTIONS_FETCH_SUCCESS,
|
|
|
|
SUGGESTIONS_FETCH_FAIL,
|
|
|
|
SUGGESTIONS_DISMISS,
|
|
|
|
SUGGESTIONS_V2_FETCH_REQUEST,
|
|
|
|
SUGGESTIONS_V2_FETCH_SUCCESS,
|
|
|
|
SUGGESTIONS_V2_FETCH_FAIL,
|
2022-07-01 13:09:07 -07:00
|
|
|
SUGGESTIONS_TRUTH_FETCH_REQUEST,
|
|
|
|
SUGGESTIONS_TRUTH_FETCH_SUCCESS,
|
|
|
|
SUGGESTIONS_TRUTH_FETCH_FAIL,
|
2022-06-15 13:11:36 -07:00
|
|
|
fetchSuggestionsV1,
|
|
|
|
fetchSuggestionsV2,
|
|
|
|
fetchSuggestions,
|
2022-07-01 13:09:07 -07:00
|
|
|
fetchSuggestionsForTimeline,
|
2022-06-15 13:11:36 -07:00
|
|
|
dismissSuggestion,
|
|
|
|
};
|