frontend-rw #1
10 changed files with 148 additions and 97 deletions
|
@ -10,7 +10,7 @@ const tokenSchema = v.object({
|
|||
scope: v.string(),
|
||||
created_at: v.fallback(v.optional(v.number()), undefined),
|
||||
|
||||
id: v.fallback(v.optional(v.number()), undefined),
|
||||
id: v.fallback(v.optional(v.pipe(v.unknown(), v.transform(String))), undefined),
|
||||
refresh_token: v.fallback(v.optional(v.string()), undefined),
|
||||
expires_in: v.fallback(v.optional(v.number()), undefined),
|
||||
me: v.fallback(v.optional(v.string()), undefined),
|
||||
|
|
|
@ -9,9 +9,9 @@
|
|||
import {
|
||||
credentialAccountSchema,
|
||||
PlApiClient,
|
||||
type Application,
|
||||
type CreateAccountParams,
|
||||
type CredentialAccount,
|
||||
type CredentialApplication,
|
||||
type Token,
|
||||
} from 'pl-api';
|
||||
import { defineMessages } from 'react-intl';
|
||||
|
@ -75,7 +75,7 @@ const createAppAndToken = () =>
|
|||
|
||||
interface AuthAppCreatedAction {
|
||||
type: typeof AUTH_APP_CREATED;
|
||||
app: Application;
|
||||
app: CredentialApplication;
|
||||
}
|
||||
|
||||
/** Create an auth app, or use it from build config */
|
||||
|
@ -104,7 +104,7 @@ const createAuthApp = () =>
|
|||
|
||||
interface AuthAppAuthorizedAction {
|
||||
type: typeof AUTH_APP_AUTHORIZED;
|
||||
app: Application;
|
||||
app: CredentialApplication;
|
||||
token: Token;
|
||||
}
|
||||
|
||||
|
@ -295,18 +295,19 @@ const logOut = () =>
|
|||
|
||||
interface SwitchAccountAction {
|
||||
type: typeof SWITCH_ACCOUNT;
|
||||
account?: Account;
|
||||
background: boolean;
|
||||
account: Account;
|
||||
}
|
||||
|
||||
const switchAccount = (accountId: string, background = false) =>
|
||||
const switchAccount = (accountId: string) =>
|
||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
||||
const account = selectAccount(getState(), accountId);
|
||||
if (!account) return;
|
||||
|
||||
// Clear all stored cache from React Query
|
||||
queryClient.invalidateQueries();
|
||||
queryClient.clear();
|
||||
|
||||
return dispatch<SwitchAccountAction>({ type: SWITCH_ACCOUNT, account, background });
|
||||
return dispatch<SwitchAccountAction>({ type: SWITCH_ACCOUNT, account });
|
||||
};
|
||||
|
||||
const fetchOwnAccounts = () =>
|
||||
|
|
|
@ -436,7 +436,7 @@ const initEventEdit = (statusId: string) => (dispatch: AppDispatch, getState: ()
|
|||
|
||||
return getClient(getState()).statuses.getStatusSource(statusId).then(response => {
|
||||
dispatch({ type: STATUS_FETCH_SOURCE_SUCCESS, statusId });
|
||||
dispatch({
|
||||
dispatch<EventFormSetAction>({
|
||||
type: EVENT_FORM_SET,
|
||||
composeId: `compose-event-modal-${statusId}`,
|
||||
text: response.text,
|
||||
|
@ -453,19 +453,19 @@ const fetchRecentEvents = () =>
|
|||
return;
|
||||
}
|
||||
|
||||
dispatch({ type: RECENT_EVENTS_FETCH_REQUEST });
|
||||
dispatch<EventsAction>({ type: RECENT_EVENTS_FETCH_REQUEST });
|
||||
|
||||
return getClient(getState()).timelines.publicTimeline({
|
||||
only_events: true,
|
||||
}).then(response => {
|
||||
dispatch(importEntities({ statuses: response.items }));
|
||||
dispatch({
|
||||
dispatch<EventsAction>({
|
||||
type: RECENT_EVENTS_FETCH_SUCCESS,
|
||||
statuses: response.items,
|
||||
next: response.next,
|
||||
});
|
||||
}).catch(error => {
|
||||
dispatch({ type: RECENT_EVENTS_FETCH_FAIL, error });
|
||||
dispatch<EventsAction>({ type: RECENT_EVENTS_FETCH_FAIL, error });
|
||||
});
|
||||
};
|
||||
|
||||
|
@ -475,17 +475,17 @@ const fetchJoinedEvents = () =>
|
|||
return;
|
||||
}
|
||||
|
||||
dispatch({ type: JOINED_EVENTS_FETCH_REQUEST });
|
||||
dispatch<EventsAction>({ type: JOINED_EVENTS_FETCH_REQUEST });
|
||||
|
||||
getClient(getState).events.getJoinedEvents().then(response => {
|
||||
dispatch(importEntities({ statuses: response.items }));
|
||||
dispatch({
|
||||
dispatch<EventsAction>({
|
||||
type: JOINED_EVENTS_FETCH_SUCCESS,
|
||||
statuses: response.items,
|
||||
next: response.next,
|
||||
});
|
||||
}).catch(error => {
|
||||
dispatch({ type: JOINED_EVENTS_FETCH_FAIL, error });
|
||||
dispatch<EventsAction>({ type: JOINED_EVENTS_FETCH_FAIL, error });
|
||||
});
|
||||
};
|
||||
|
||||
|
@ -520,7 +520,13 @@ type EventsAction =
|
|||
| ReturnType<typeof rejectEventParticipationRequestSuccess>
|
||||
| ReturnType<typeof rejectEventParticipationRequestFail>
|
||||
| ReturnType<typeof cancelEventCompose>
|
||||
| EventFormSetAction;
|
||||
| EventFormSetAction
|
||||
| { type: typeof RECENT_EVENTS_FETCH_REQUEST }
|
||||
| { type: typeof RECENT_EVENTS_FETCH_SUCCESS; statuses: Array<Status>; next: (() => Promise<PaginatedResponse<Status>>) | null }
|
||||
| { type: typeof RECENT_EVENTS_FETCH_FAIL; error: unknown }
|
||||
| { type: typeof JOINED_EVENTS_FETCH_REQUEST }
|
||||
| { type: typeof JOINED_EVENTS_FETCH_SUCCESS; statuses: Array<Status>; next: (() => Promise<PaginatedResponse<Status>>) | null }
|
||||
| { type: typeof JOINED_EVENTS_FETCH_FAIL; error: unknown }
|
||||
|
||||
export {
|
||||
EVENT_SUBMIT_REQUEST,
|
||||
|
|
|
@ -1,5 +1,6 @@
|
|||
import { getClient } from '../api';
|
||||
|
||||
import type { PlApiClient } from 'pl-api';
|
||||
import type { AppDispatch, RootState } from 'pl-fe/store';
|
||||
|
||||
const MFA_FETCH_REQUEST = 'MFA_FETCH_REQUEST' as const;
|
||||
|
@ -24,62 +25,79 @@ const MFA_DISABLE_FAIL = 'MFA_DISABLE_FAIL' as const;
|
|||
|
||||
const fetchMfa = () =>
|
||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
||||
dispatch({ type: MFA_FETCH_REQUEST });
|
||||
dispatch<MfaAction>({ type: MFA_FETCH_REQUEST });
|
||||
return getClient(getState).settings.mfa.getMfaSettings().then((data) => {
|
||||
dispatch({ type: MFA_FETCH_SUCCESS, data });
|
||||
dispatch<MfaAction>({ type: MFA_FETCH_SUCCESS, data });
|
||||
}).catch(() => {
|
||||
dispatch({ type: MFA_FETCH_FAIL });
|
||||
dispatch<MfaAction>({ type: MFA_FETCH_FAIL });
|
||||
});
|
||||
};
|
||||
|
||||
const fetchBackupCodes = () =>
|
||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
||||
dispatch({ type: MFA_BACKUP_CODES_FETCH_REQUEST });
|
||||
dispatch<MfaAction>({ type: MFA_BACKUP_CODES_FETCH_REQUEST });
|
||||
return getClient(getState).settings.mfa.getMfaBackupCodes().then((data) => {
|
||||
dispatch({ type: MFA_BACKUP_CODES_FETCH_SUCCESS, data });
|
||||
dispatch<MfaAction>({ type: MFA_BACKUP_CODES_FETCH_SUCCESS, data });
|
||||
return data;
|
||||
}).catch((error: unknown) => {
|
||||
dispatch({ type: MFA_BACKUP_CODES_FETCH_FAIL });
|
||||
dispatch<MfaAction>({ type: MFA_BACKUP_CODES_FETCH_FAIL });
|
||||
throw error;
|
||||
});
|
||||
};
|
||||
|
||||
const setupMfa = (method: 'totp') =>
|
||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
||||
dispatch({ type: MFA_SETUP_REQUEST, method });
|
||||
dispatch<MfaAction>({ type: MFA_SETUP_REQUEST, method });
|
||||
return getClient(getState).settings.mfa.getMfaSetup(method).then((data) => {
|
||||
dispatch({ type: MFA_SETUP_SUCCESS, data });
|
||||
dispatch<MfaAction>({ type: MFA_SETUP_SUCCESS, data });
|
||||
return data;
|
||||
}).catch((error: unknown) => {
|
||||
dispatch({ type: MFA_SETUP_FAIL });
|
||||
dispatch<MfaAction>({ type: MFA_SETUP_FAIL });
|
||||
throw error;
|
||||
});
|
||||
};
|
||||
|
||||
const confirmMfa = (method: 'totp', code: string, password: string) =>
|
||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
||||
dispatch({ type: MFA_CONFIRM_REQUEST, method, code });
|
||||
dispatch<MfaAction>({ type: MFA_CONFIRM_REQUEST, method, code });
|
||||
return getClient(getState).settings.mfa.confirmMfaSetup(method, code, password).then((data) => {
|
||||
dispatch({ type: MFA_CONFIRM_SUCCESS, method, code });
|
||||
dispatch<MfaAction>({ type: MFA_CONFIRM_SUCCESS, method, code });
|
||||
return data;
|
||||
}).catch((error: unknown) => {
|
||||
dispatch({ type: MFA_CONFIRM_FAIL, method, code, error, skipAlert: true });
|
||||
dispatch<MfaAction>({ type: MFA_CONFIRM_FAIL, method, code, error, skipAlert: true });
|
||||
throw error;
|
||||
});
|
||||
};
|
||||
|
||||
const disableMfa = (method: 'totp', password: string) =>
|
||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
||||
dispatch({ type: MFA_DISABLE_REQUEST, method });
|
||||
dispatch<MfaAction>({ type: MFA_DISABLE_REQUEST, method });
|
||||
return getClient(getState).settings.mfa.disableMfa(method, password).then((data) => {
|
||||
dispatch({ type: MFA_DISABLE_SUCCESS, method });
|
||||
dispatch<MfaAction>({ type: MFA_DISABLE_SUCCESS, method });
|
||||
return data;
|
||||
}).catch((error: unknown) => {
|
||||
dispatch({ type: MFA_DISABLE_FAIL, method, skipAlert: true });
|
||||
dispatch<MfaAction>({ type: MFA_DISABLE_FAIL, method, skipAlert: true });
|
||||
throw error;
|
||||
});
|
||||
};
|
||||
|
||||
type MfaAction =
|
||||
| { type: typeof MFA_FETCH_REQUEST }
|
||||
| { type: typeof MFA_FETCH_SUCCESS; data: Awaited<ReturnType<(InstanceType<typeof PlApiClient>)['settings']['mfa']['getMfaSettings']>> }
|
||||
| { type: typeof MFA_FETCH_FAIL }
|
||||
| { type: typeof MFA_BACKUP_CODES_FETCH_REQUEST }
|
||||
| { type: typeof MFA_BACKUP_CODES_FETCH_SUCCESS; data: Awaited<ReturnType<(InstanceType<typeof PlApiClient>)['settings']['mfa']['getMfaBackupCodes']>> }
|
||||
| { type: typeof MFA_BACKUP_CODES_FETCH_FAIL }
|
||||
| { type: typeof MFA_SETUP_REQUEST; method: 'totp' }
|
||||
| { type: typeof MFA_SETUP_SUCCESS; data: Awaited<ReturnType<(InstanceType<typeof PlApiClient>)['settings']['mfa']['getMfaSetup']>> }
|
||||
| { type: typeof MFA_SETUP_FAIL }
|
||||
| { type: typeof MFA_CONFIRM_REQUEST; method: 'totp'; code: string }
|
||||
| { type: typeof MFA_CONFIRM_SUCCESS; method: 'totp'; code: string }
|
||||
| { type: typeof MFA_CONFIRM_FAIL; method: 'totp'; code: string; error: unknown; skipAlert: true }
|
||||
| { type: typeof MFA_DISABLE_REQUEST; method: 'totp' }
|
||||
| { type: typeof MFA_DISABLE_SUCCESS; method: 'totp' }
|
||||
| { type: typeof MFA_DISABLE_FAIL; method: 'totp'; skipAlert: true };
|
||||
|
||||
export {
|
||||
MFA_FETCH_REQUEST,
|
||||
MFA_FETCH_SUCCESS,
|
||||
|
@ -101,4 +119,5 @@ export {
|
|||
setupMfa,
|
||||
confirmMfa,
|
||||
disableMfa,
|
||||
type MfaAction,
|
||||
};
|
||||
|
|
|
@ -4,7 +4,7 @@ import { getClient } from '../api';
|
|||
|
||||
import { importEntities } from './importer';
|
||||
|
||||
import type { Status } from 'pl-api';
|
||||
import type { PaginatedResponse, Status } from 'pl-api';
|
||||
import type { AppDispatch, RootState } from 'pl-fe/store';
|
||||
|
||||
const PINNED_STATUSES_FETCH_REQUEST = 'PINNED_STATUSES_FETCH_REQUEST' as const;
|
||||
|
@ -20,7 +20,7 @@ const fetchPinnedStatuses = () =>
|
|||
|
||||
return getClient(getState()).accounts.getAccountStatuses(me as string, { pinned: true }).then(response => {
|
||||
dispatch(importEntities({ statuses: response.items }));
|
||||
dispatch(fetchPinnedStatusesSuccess(response.items, null));
|
||||
dispatch(fetchPinnedStatusesSuccess(response.items, response.next));
|
||||
}).catch(error => {
|
||||
dispatch(fetchPinnedStatusesFail(error));
|
||||
});
|
||||
|
@ -30,7 +30,7 @@ const fetchPinnedStatusesRequest = () => ({
|
|||
type: PINNED_STATUSES_FETCH_REQUEST,
|
||||
});
|
||||
|
||||
const fetchPinnedStatusesSuccess = (statuses: Array<Status>, next: string | null) => ({
|
||||
const fetchPinnedStatusesSuccess = (statuses: Array<Status>, next: (() => Promise<PaginatedResponse<Status>>) | null) => ({
|
||||
type: PINNED_STATUSES_FETCH_SUCCESS,
|
||||
statuses,
|
||||
next,
|
||||
|
|
|
@ -11,6 +11,8 @@ import { normalizeUsername } from 'pl-fe/utils/input';
|
|||
|
||||
import { AUTH_LOGGED_OUT, messages } from './auth';
|
||||
|
||||
import type { OauthToken } from 'pl-api';
|
||||
import type { Account } from 'pl-fe/normalizers/account';
|
||||
import type { AppDispatch, RootState } from 'pl-fe/store';
|
||||
|
||||
const FETCH_TOKENS_REQUEST = 'FETCH_TOKENS_REQUEST' as const;
|
||||
|
@ -47,32 +49,32 @@ const MOVE_ACCOUNT_FAIL = 'MOVE_ACCOUNT_FAIL' as const;
|
|||
|
||||
const fetchOAuthTokens = () =>
|
||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
||||
dispatch({ type: FETCH_TOKENS_REQUEST });
|
||||
dispatch<SecurityAction>({ type: FETCH_TOKENS_REQUEST });
|
||||
return getClient(getState).settings.getOauthTokens().then((tokens) => {
|
||||
dispatch({ type: FETCH_TOKENS_SUCCESS, tokens });
|
||||
dispatch<SecurityAction>({ type: FETCH_TOKENS_SUCCESS, tokens });
|
||||
}).catch((e) => {
|
||||
dispatch({ type: FETCH_TOKENS_FAIL });
|
||||
dispatch<SecurityAction>({ type: FETCH_TOKENS_FAIL });
|
||||
});
|
||||
};
|
||||
|
||||
const revokeOAuthTokenById = (tokenId: number) =>
|
||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
||||
dispatch({ type: REVOKE_TOKEN_REQUEST, tokenId });
|
||||
dispatch<SecurityAction>({ type: REVOKE_TOKEN_REQUEST, tokenId });
|
||||
return getClient(getState).settings.deleteOauthToken(tokenId).then(() => {
|
||||
dispatch({ type: REVOKE_TOKEN_SUCCESS, tokenId });
|
||||
dispatch<SecurityAction>({ type: REVOKE_TOKEN_SUCCESS, tokenId });
|
||||
}).catch(() => {
|
||||
dispatch({ type: REVOKE_TOKEN_FAIL, tokenId });
|
||||
dispatch<SecurityAction>({ type: REVOKE_TOKEN_FAIL, tokenId });
|
||||
});
|
||||
};
|
||||
|
||||
const changePassword = (oldPassword: string, newPassword: string) =>
|
||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
||||
dispatch({ type: CHANGE_PASSWORD_REQUEST });
|
||||
dispatch<SecurityAction>({ type: CHANGE_PASSWORD_REQUEST });
|
||||
|
||||
return getClient(getState).settings.changePassword(oldPassword, newPassword).then(response => {
|
||||
dispatch({ type: CHANGE_PASSWORD_SUCCESS, response });
|
||||
dispatch<SecurityAction>({ type: CHANGE_PASSWORD_SUCCESS, response });
|
||||
}).catch(error => {
|
||||
dispatch({ type: CHANGE_PASSWORD_FAIL, error, skipAlert: true });
|
||||
dispatch<SecurityAction>({ type: CHANGE_PASSWORD_FAIL, error, skipAlert: true });
|
||||
throw error;
|
||||
});
|
||||
};
|
||||
|
@ -81,58 +83,83 @@ const resetPassword = (usernameOrEmail: string) =>
|
|||
(dispatch: AppDispatch, getState: () => RootState) => {
|
||||
const input = normalizeUsername(usernameOrEmail);
|
||||
|
||||
dispatch({ type: RESET_PASSWORD_REQUEST });
|
||||
dispatch<SecurityAction>({ type: RESET_PASSWORD_REQUEST });
|
||||
|
||||
return getClient(getState).settings.resetPassword(
|
||||
input.includes('@') ? input : undefined,
|
||||
input.includes('@') ? undefined : input,
|
||||
).then(() => {
|
||||
dispatch({ type: RESET_PASSWORD_SUCCESS });
|
||||
dispatch<SecurityAction>({ type: RESET_PASSWORD_SUCCESS });
|
||||
}).catch(error => {
|
||||
dispatch({ type: RESET_PASSWORD_FAIL, error });
|
||||
dispatch<SecurityAction>({ type: RESET_PASSWORD_FAIL, error });
|
||||
throw error;
|
||||
});
|
||||
};
|
||||
|
||||
const changeEmail = (email: string, password: string) =>
|
||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
||||
dispatch({ type: CHANGE_EMAIL_REQUEST, email });
|
||||
dispatch<SecurityAction>({ type: CHANGE_EMAIL_REQUEST, email });
|
||||
|
||||
return getClient(getState).settings.changeEmail(email, password).then(response => {
|
||||
dispatch({ type: CHANGE_EMAIL_SUCCESS, email, response });
|
||||
dispatch<SecurityAction>({ type: CHANGE_EMAIL_SUCCESS, email, response });
|
||||
}).catch(error => {
|
||||
dispatch({ type: CHANGE_EMAIL_FAIL, email, error, skipAlert: true });
|
||||
dispatch<SecurityAction>({ type: CHANGE_EMAIL_FAIL, email, error, skipAlert: true });
|
||||
throw error;
|
||||
});
|
||||
};
|
||||
|
||||
const deleteAccount = (password: string) =>
|
||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
||||
dispatch({ type: CHANGE_PASSWORD_REQUEST });
|
||||
const account = getLoggedInAccount(getState());
|
||||
dispatch<SecurityAction>({ type: CHANGE_PASSWORD_REQUEST });
|
||||
const account = getLoggedInAccount(getState())!;
|
||||
|
||||
dispatch({ type: DELETE_ACCOUNT_REQUEST });
|
||||
dispatch<SecurityAction>({ type: DELETE_ACCOUNT_REQUEST });
|
||||
return getClient(getState).settings.deleteAccount(password).then(response => {
|
||||
dispatch({ type: DELETE_ACCOUNT_SUCCESS, response });
|
||||
dispatch({ type: AUTH_LOGGED_OUT, account });
|
||||
dispatch<SecurityAction>({ type: DELETE_ACCOUNT_SUCCESS, response });
|
||||
dispatch<SecurityAction>({ type: AUTH_LOGGED_OUT, account });
|
||||
toast.success(messages.loggedOut);
|
||||
}).catch(error => {
|
||||
dispatch({ type: DELETE_ACCOUNT_FAIL, error, skipAlert: true });
|
||||
dispatch<SecurityAction>({ type: DELETE_ACCOUNT_FAIL, error, skipAlert: true });
|
||||
throw error;
|
||||
});
|
||||
};
|
||||
|
||||
const moveAccount = (targetAccount: string, password: string) =>
|
||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
||||
dispatch({ type: MOVE_ACCOUNT_REQUEST });
|
||||
dispatch<SecurityAction>({ type: MOVE_ACCOUNT_REQUEST });
|
||||
return getClient(getState).settings.moveAccount(targetAccount, password).then(response => {
|
||||
dispatch({ type: MOVE_ACCOUNT_SUCCESS, response });
|
||||
dispatch<SecurityAction>({ type: MOVE_ACCOUNT_SUCCESS, response });
|
||||
}).catch(error => {
|
||||
dispatch({ type: MOVE_ACCOUNT_FAIL, error, skipAlert: true });
|
||||
dispatch<SecurityAction>({ type: MOVE_ACCOUNT_FAIL, error, skipAlert: true });
|
||||
throw error;
|
||||
});
|
||||
};
|
||||
|
||||
type SecurityAction =
|
||||
| { type: typeof FETCH_TOKENS_REQUEST }
|
||||
| { type: typeof FETCH_TOKENS_SUCCESS; tokens: Array<OauthToken> }
|
||||
| { type: typeof FETCH_TOKENS_FAIL }
|
||||
| { type: typeof REVOKE_TOKEN_REQUEST; tokenId: number }
|
||||
| { type: typeof REVOKE_TOKEN_SUCCESS; tokenId: number }
|
||||
| { type: typeof REVOKE_TOKEN_FAIL; tokenId: number }
|
||||
| { type: typeof CHANGE_PASSWORD_REQUEST }
|
||||
| { type: typeof CHANGE_PASSWORD_SUCCESS; response: {} }
|
||||
| { type: typeof CHANGE_PASSWORD_FAIL; error: unknown; skipAlert: true }
|
||||
| { type: typeof RESET_PASSWORD_REQUEST }
|
||||
| { type: typeof RESET_PASSWORD_SUCCESS }
|
||||
| { type: typeof RESET_PASSWORD_FAIL; error: unknown }
|
||||
| { type: typeof CHANGE_EMAIL_REQUEST; email: string }
|
||||
| { type: typeof CHANGE_EMAIL_SUCCESS; email: string; response: {} }
|
||||
| { type: typeof CHANGE_EMAIL_FAIL; email: string; error: unknown; skipAlert: true }
|
||||
| { type: typeof CHANGE_PASSWORD_REQUEST }
|
||||
| { type: typeof DELETE_ACCOUNT_REQUEST }
|
||||
| { type: typeof DELETE_ACCOUNT_SUCCESS; response: {} }
|
||||
| { type: typeof AUTH_LOGGED_OUT; account: Account }
|
||||
| { type: typeof DELETE_ACCOUNT_FAIL; error: unknown; skipAlert: true }
|
||||
| { type: typeof MOVE_ACCOUNT_REQUEST }
|
||||
| { type: typeof MOVE_ACCOUNT_SUCCESS; response: {} }
|
||||
| { type: typeof MOVE_ACCOUNT_FAIL; error: unknown; skipAlert: true }
|
||||
|
||||
export {
|
||||
FETCH_TOKENS_REQUEST,
|
||||
FETCH_TOKENS_SUCCESS,
|
||||
|
@ -165,4 +192,5 @@ export {
|
|||
changeEmail,
|
||||
deleteAccount,
|
||||
moveAccount,
|
||||
type SecurityAction,
|
||||
};
|
||||
|
|
|
@ -23,7 +23,8 @@ import { ME_FETCH_SKIP, type MeAction } from '../actions/me';
|
|||
|
||||
import type { PlfeResponse } from 'pl-fe/api';
|
||||
import type { Account as AccountEntity } from 'pl-fe/normalizers/account';
|
||||
import type { AnyAction } from 'redux';
|
||||
|
||||
type Action = AuthAction | MeAction | PreloadAction;
|
||||
|
||||
const backendUrl = (isURL(BuildConfig.BACKEND_URL) ? BuildConfig.BACKEND_URL : '');
|
||||
|
||||
|
@ -296,7 +297,7 @@ const updateState = (state: State, updater: (state: Draft<State>) => void, clien
|
|||
return { ...newState, client: newClient };
|
||||
};
|
||||
|
||||
const reducer = (state: State, action: AnyAction | AuthAction | MeAction | PreloadAction): State => {
|
||||
const reducer = (state: State, action: Action): State => {
|
||||
switch (action.type) {
|
||||
case AUTH_APP_CREATED:
|
||||
return updateState(state, (draft) => {
|
||||
|
@ -304,7 +305,7 @@ const reducer = (state: State, action: AnyAction | AuthAction | MeAction | Prelo
|
|||
});
|
||||
case AUTH_APP_AUTHORIZED:
|
||||
return updateState(state, (draft) => {
|
||||
draft.app = ({ ...draft.app, ...action.token });
|
||||
if (draft.app) draft.app = ({ ...draft.app, ...action.token });
|
||||
});
|
||||
case AUTH_LOGGED_IN:
|
||||
return updateState(state, (draft) => {
|
||||
|
@ -328,17 +329,20 @@ const reducer = (state: State, action: AnyAction | AuthAction | MeAction | Prelo
|
|||
});
|
||||
case VERIFY_CREDENTIALS_FAIL:
|
||||
return updateState(state, (draft) => {
|
||||
deleteForbiddenToken(draft, action.error, action.token);
|
||||
deleteForbiddenToken(draft, action.error as any, action.token);
|
||||
});
|
||||
case SWITCH_ACCOUNT:
|
||||
return updateState(state, (draft) => {
|
||||
draft.me = action.account.url;
|
||||
}, () => {
|
||||
const accessToken = state.users[action.account.url]?.access_token;
|
||||
|
||||
if (state.client.baseURL === parseBaseURL(action.account.url)) {
|
||||
state.client.accessToken = action.account.access_token;
|
||||
state.client.accessToken = accessToken;
|
||||
return state.client;
|
||||
}
|
||||
return new PlApiClient(parseBaseURL(action.account.url) || backendUrl, action.account.access_token);
|
||||
|
||||
return new PlApiClient(parseBaseURL(action.account.url) || backendUrl, accessToken);
|
||||
});
|
||||
case ME_FETCH_SKIP:
|
||||
return updateState(state, (draft) => {
|
||||
|
@ -373,7 +377,7 @@ const userSwitched = (oldState: State, state: State) => {
|
|||
return stillValid && didChange && !userUpgradedUrl;
|
||||
};
|
||||
|
||||
const maybeReload = (oldState: State, state: State, action: AnyAction) => {
|
||||
const maybeReload = (oldState: State, state: State, action: Action) => {
|
||||
const loggedOutStandalone = action.type === AUTH_LOGGED_OUT && action.standalone;
|
||||
const switched = userSwitched(oldState, state);
|
||||
|
||||
|
@ -382,19 +386,13 @@ const maybeReload = (oldState: State, state: State, action: AnyAction) => {
|
|||
}
|
||||
};
|
||||
|
||||
const auth = (oldState: State = initialState, action: AnyAction): State => {
|
||||
const auth = (oldState: State = initialState, action: Action): State => {
|
||||
const state = reducer(oldState, action);
|
||||
|
||||
if (state !== oldState) {
|
||||
// Persist the state in localStorage
|
||||
persistAuth(state);
|
||||
|
||||
// When middle-clicking a profile, we want to save the
|
||||
// user in localStorage, but not update the reducer
|
||||
if (action.background === true) {
|
||||
return oldState;
|
||||
}
|
||||
|
||||
// Persist the session
|
||||
persistSession(state);
|
||||
|
||||
|
|
|
@ -77,7 +77,7 @@ const persistInstance = (instance: { domain: string }, host: string | null = get
|
|||
}
|
||||
};
|
||||
|
||||
const handleInstanceFetchFail = (state: State, error: Record<string, any>) => {
|
||||
const handleInstanceFetchFail = (state: State, error: any) => {
|
||||
if (error.response?.status === 401) {
|
||||
return handleAuthFetch(state);
|
||||
} else {
|
||||
|
|
|
@ -4,11 +4,11 @@ import {
|
|||
MFA_FETCH_SUCCESS,
|
||||
MFA_CONFIRM_SUCCESS,
|
||||
MFA_DISABLE_SUCCESS,
|
||||
type MfaAction,
|
||||
} from '../actions/mfa';
|
||||
import { FETCH_TOKENS_SUCCESS, REVOKE_TOKEN_SUCCESS } from '../actions/security';
|
||||
import { FETCH_TOKENS_SUCCESS, REVOKE_TOKEN_SUCCESS, type SecurityAction } from '../actions/security';
|
||||
|
||||
import type { OauthToken } from 'pl-api';
|
||||
import type { AnyAction } from 'redux';
|
||||
|
||||
interface State {
|
||||
tokens: Array<OauthToken>;
|
||||
|
@ -34,7 +34,7 @@ const enableMfa = (state: State, method: string) => state.mfa.settings = { ...st
|
|||
|
||||
const disableMfa = (state: State, method: string) => state.mfa.settings = { ...state.mfa.settings, [method]: false };
|
||||
|
||||
const security = (state = initialState, action: AnyAction) => {
|
||||
const security = (state = initialState, action: MfaAction | SecurityAction) => {
|
||||
switch (action.type) {
|
||||
case FETCH_TOKENS_SUCCESS:
|
||||
return create(state, (draft) => {
|
||||
|
|
|
@ -1,16 +1,5 @@
|
|||
import { create } from 'mutative';
|
||||
|
||||
import {
|
||||
STATUS_QUOTES_EXPAND_FAIL,
|
||||
STATUS_QUOTES_EXPAND_REQUEST,
|
||||
STATUS_QUOTES_EXPAND_SUCCESS,
|
||||
STATUS_QUOTES_FETCH_FAIL,
|
||||
STATUS_QUOTES_FETCH_REQUEST,
|
||||
STATUS_QUOTES_FETCH_SUCCESS,
|
||||
type StatusQuotesAction,
|
||||
} from 'pl-fe/actions/status-quotes';
|
||||
import { STATUS_CREATE_SUCCESS } from 'pl-fe/actions/statuses';
|
||||
|
||||
import {
|
||||
BOOKMARKED_STATUSES_FETCH_REQUEST,
|
||||
BOOKMARKED_STATUSES_FETCH_SUCCESS,
|
||||
|
@ -19,7 +8,7 @@ import {
|
|||
BOOKMARKED_STATUSES_EXPAND_SUCCESS,
|
||||
BOOKMARKED_STATUSES_EXPAND_FAIL,
|
||||
type BookmarksAction,
|
||||
} from '../actions/bookmarks';
|
||||
} from 'pl-fe/actions/bookmarks';
|
||||
import {
|
||||
RECENT_EVENTS_FETCH_REQUEST,
|
||||
RECENT_EVENTS_FETCH_SUCCESS,
|
||||
|
@ -28,7 +17,7 @@ import {
|
|||
JOINED_EVENTS_FETCH_SUCCESS,
|
||||
JOINED_EVENTS_FETCH_FAIL,
|
||||
type EventsAction,
|
||||
} from '../actions/events';
|
||||
} from 'pl-fe/actions/events';
|
||||
import {
|
||||
FAVOURITED_STATUSES_FETCH_REQUEST,
|
||||
FAVOURITED_STATUSES_FETCH_SUCCESS,
|
||||
|
@ -43,7 +32,7 @@ import {
|
|||
ACCOUNT_FAVOURITED_STATUSES_EXPAND_SUCCESS,
|
||||
ACCOUNT_FAVOURITED_STATUSES_EXPAND_FAIL,
|
||||
type FavouritesAction,
|
||||
} from '../actions/favourites';
|
||||
} from 'pl-fe/actions/favourites';
|
||||
import {
|
||||
FAVOURITE_SUCCESS,
|
||||
UNFAVOURITE_SUCCESS,
|
||||
|
@ -52,8 +41,8 @@ import {
|
|||
PIN_SUCCESS,
|
||||
UNPIN_SUCCESS,
|
||||
type InteractionsAction,
|
||||
} from '../actions/interactions';
|
||||
import { PINNED_STATUSES_FETCH_SUCCESS, type PinStatusesAction } from '../actions/pin-statuses';
|
||||
} from 'pl-fe/actions/interactions';
|
||||
import { PINNED_STATUSES_FETCH_SUCCESS, type PinStatusesAction } from 'pl-fe/actions/pin-statuses';
|
||||
import {
|
||||
SCHEDULED_STATUSES_FETCH_REQUEST,
|
||||
SCHEDULED_STATUSES_FETCH_SUCCESS,
|
||||
|
@ -63,10 +52,20 @@ import {
|
|||
SCHEDULED_STATUSES_EXPAND_FAIL,
|
||||
SCHEDULED_STATUS_CANCEL_REQUEST,
|
||||
SCHEDULED_STATUS_CANCEL_SUCCESS,
|
||||
} from '../actions/scheduled-statuses';
|
||||
type ScheduledStatusesAction,
|
||||
} from 'pl-fe/actions/scheduled-statuses';
|
||||
import {
|
||||
STATUS_QUOTES_EXPAND_FAIL,
|
||||
STATUS_QUOTES_EXPAND_REQUEST,
|
||||
STATUS_QUOTES_EXPAND_SUCCESS,
|
||||
STATUS_QUOTES_FETCH_FAIL,
|
||||
STATUS_QUOTES_FETCH_REQUEST,
|
||||
STATUS_QUOTES_FETCH_SUCCESS,
|
||||
type StatusQuotesAction,
|
||||
} from 'pl-fe/actions/status-quotes';
|
||||
import { STATUS_CREATE_SUCCESS, type StatusesAction } from 'pl-fe/actions/statuses';
|
||||
|
||||
import type { PaginatedResponse, ScheduledStatus, Status } from 'pl-api';
|
||||
import type { AnyAction } from 'redux';
|
||||
|
||||
interface StatusList {
|
||||
next: (() => Promise<PaginatedResponse<Status>>) | null;
|
||||
|
@ -156,7 +155,7 @@ const removeBookmarkFromLists = (state: State, status: Pick<Status, 'id' | 'book
|
|||
}
|
||||
};
|
||||
|
||||
const statusLists = (state = initialState, action: AnyAction | BookmarksAction | EventsAction | FavouritesAction | InteractionsAction | PinStatusesAction | StatusQuotesAction): State => {
|
||||
const statusLists = (state = initialState, action: BookmarksAction | EventsAction | FavouritesAction | InteractionsAction | PinStatusesAction | ScheduledStatusesAction | StatusesAction | StatusQuotesAction): State => {
|
||||
switch (action.type) {
|
||||
case FAVOURITED_STATUSES_FETCH_REQUEST:
|
||||
case FAVOURITED_STATUSES_EXPAND_REQUEST:
|
||||
|
@ -209,9 +208,9 @@ const statusLists = (state = initialState, action: AnyAction | BookmarksAction |
|
|||
case SCHEDULED_STATUSES_EXPAND_FAIL:
|
||||
return create(state, draft => setLoading(draft, 'scheduled_statuses', false));
|
||||
case SCHEDULED_STATUSES_FETCH_SUCCESS:
|
||||
return create(state, draft => normalizeList(draft, 'scheduled_statuses', action.statuses, action.next));
|
||||
return create(state, draft => normalizeList(draft, 'scheduled_statuses', action.statuses, action.next as any));
|
||||
case SCHEDULED_STATUSES_EXPAND_SUCCESS:
|
||||
return create(state, draft => appendToList(draft, 'scheduled_statuses', action.statuses, action.next));
|
||||
return create(state, draft => appendToList(draft, 'scheduled_statuses', action.statuses, action.next as any));
|
||||
case SCHEDULED_STATUS_CANCEL_REQUEST:
|
||||
case SCHEDULED_STATUS_CANCEL_SUCCESS:
|
||||
return create(state, draft => removeOneFromList(draft, 'scheduled_statuses', action.statusId));
|
||||
|
|
Loading…
Reference in a new issue