pl-fe: types

Signed-off-by: marcin mikołajczak <git@mkljczk.pl>
This commit is contained in:
marcin mikołajczak 2024-11-26 18:46:44 +01:00
parent 5e1d0faabc
commit dddac128fb
10 changed files with 148 additions and 97 deletions

View file

@ -10,7 +10,7 @@ const tokenSchema = v.object({
scope: v.string(), scope: v.string(),
created_at: v.fallback(v.optional(v.number()), undefined), 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), refresh_token: v.fallback(v.optional(v.string()), undefined),
expires_in: v.fallback(v.optional(v.number()), undefined), expires_in: v.fallback(v.optional(v.number()), undefined),
me: v.fallback(v.optional(v.string()), undefined), me: v.fallback(v.optional(v.string()), undefined),

View file

@ -9,9 +9,9 @@
import { import {
credentialAccountSchema, credentialAccountSchema,
PlApiClient, PlApiClient,
type Application,
type CreateAccountParams, type CreateAccountParams,
type CredentialAccount, type CredentialAccount,
type CredentialApplication,
type Token, type Token,
} from 'pl-api'; } from 'pl-api';
import { defineMessages } from 'react-intl'; import { defineMessages } from 'react-intl';
@ -75,7 +75,7 @@ const createAppAndToken = () =>
interface AuthAppCreatedAction { interface AuthAppCreatedAction {
type: typeof AUTH_APP_CREATED; type: typeof AUTH_APP_CREATED;
app: Application; app: CredentialApplication;
} }
/** Create an auth app, or use it from build config */ /** Create an auth app, or use it from build config */
@ -104,7 +104,7 @@ const createAuthApp = () =>
interface AuthAppAuthorizedAction { interface AuthAppAuthorizedAction {
type: typeof AUTH_APP_AUTHORIZED; type: typeof AUTH_APP_AUTHORIZED;
app: Application; app: CredentialApplication;
token: Token; token: Token;
} }
@ -295,18 +295,19 @@ const logOut = () =>
interface SwitchAccountAction { interface SwitchAccountAction {
type: typeof SWITCH_ACCOUNT; type: typeof SWITCH_ACCOUNT;
account?: Account; account: Account;
background: boolean;
} }
const switchAccount = (accountId: string, background = false) => const switchAccount = (accountId: string) =>
(dispatch: AppDispatch, getState: () => RootState) => { (dispatch: AppDispatch, getState: () => RootState) => {
const account = selectAccount(getState(), accountId); const account = selectAccount(getState(), accountId);
if (!account) return;
// Clear all stored cache from React Query // Clear all stored cache from React Query
queryClient.invalidateQueries(); queryClient.invalidateQueries();
queryClient.clear(); queryClient.clear();
return dispatch<SwitchAccountAction>({ type: SWITCH_ACCOUNT, account, background }); return dispatch<SwitchAccountAction>({ type: SWITCH_ACCOUNT, account });
}; };
const fetchOwnAccounts = () => const fetchOwnAccounts = () =>

View file

@ -436,7 +436,7 @@ const initEventEdit = (statusId: string) => (dispatch: AppDispatch, getState: ()
return getClient(getState()).statuses.getStatusSource(statusId).then(response => { return getClient(getState()).statuses.getStatusSource(statusId).then(response => {
dispatch({ type: STATUS_FETCH_SOURCE_SUCCESS, statusId }); dispatch({ type: STATUS_FETCH_SOURCE_SUCCESS, statusId });
dispatch({ dispatch<EventFormSetAction>({
type: EVENT_FORM_SET, type: EVENT_FORM_SET,
composeId: `compose-event-modal-${statusId}`, composeId: `compose-event-modal-${statusId}`,
text: response.text, text: response.text,
@ -453,19 +453,19 @@ const fetchRecentEvents = () =>
return; return;
} }
dispatch({ type: RECENT_EVENTS_FETCH_REQUEST }); dispatch<EventsAction>({ type: RECENT_EVENTS_FETCH_REQUEST });
return getClient(getState()).timelines.publicTimeline({ return getClient(getState()).timelines.publicTimeline({
only_events: true, only_events: true,
}).then(response => { }).then(response => {
dispatch(importEntities({ statuses: response.items })); dispatch(importEntities({ statuses: response.items }));
dispatch({ dispatch<EventsAction>({
type: RECENT_EVENTS_FETCH_SUCCESS, type: RECENT_EVENTS_FETCH_SUCCESS,
statuses: response.items, statuses: response.items,
next: response.next, next: response.next,
}); });
}).catch(error => { }).catch(error => {
dispatch({ type: RECENT_EVENTS_FETCH_FAIL, error }); dispatch<EventsAction>({ type: RECENT_EVENTS_FETCH_FAIL, error });
}); });
}; };
@ -475,17 +475,17 @@ const fetchJoinedEvents = () =>
return; return;
} }
dispatch({ type: JOINED_EVENTS_FETCH_REQUEST }); dispatch<EventsAction>({ type: JOINED_EVENTS_FETCH_REQUEST });
getClient(getState).events.getJoinedEvents().then(response => { getClient(getState).events.getJoinedEvents().then(response => {
dispatch(importEntities({ statuses: response.items })); dispatch(importEntities({ statuses: response.items }));
dispatch({ dispatch<EventsAction>({
type: JOINED_EVENTS_FETCH_SUCCESS, type: JOINED_EVENTS_FETCH_SUCCESS,
statuses: response.items, statuses: response.items,
next: response.next, next: response.next,
}); });
}).catch(error => { }).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 rejectEventParticipationRequestSuccess>
| ReturnType<typeof rejectEventParticipationRequestFail> | ReturnType<typeof rejectEventParticipationRequestFail>
| ReturnType<typeof cancelEventCompose> | 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 { export {
EVENT_SUBMIT_REQUEST, EVENT_SUBMIT_REQUEST,

View file

@ -1,5 +1,6 @@
import { getClient } from '../api'; import { getClient } from '../api';
import type { PlApiClient } from 'pl-api';
import type { AppDispatch, RootState } from 'pl-fe/store'; import type { AppDispatch, RootState } from 'pl-fe/store';
const MFA_FETCH_REQUEST = 'MFA_FETCH_REQUEST' as const; const MFA_FETCH_REQUEST = 'MFA_FETCH_REQUEST' as const;
@ -24,62 +25,79 @@ const MFA_DISABLE_FAIL = 'MFA_DISABLE_FAIL' as const;
const fetchMfa = () => const fetchMfa = () =>
(dispatch: AppDispatch, getState: () => RootState) => { (dispatch: AppDispatch, getState: () => RootState) => {
dispatch({ type: MFA_FETCH_REQUEST }); dispatch<MfaAction>({ type: MFA_FETCH_REQUEST });
return getClient(getState).settings.mfa.getMfaSettings().then((data) => { return getClient(getState).settings.mfa.getMfaSettings().then((data) => {
dispatch({ type: MFA_FETCH_SUCCESS, data }); dispatch<MfaAction>({ type: MFA_FETCH_SUCCESS, data });
}).catch(() => { }).catch(() => {
dispatch({ type: MFA_FETCH_FAIL }); dispatch<MfaAction>({ type: MFA_FETCH_FAIL });
}); });
}; };
const fetchBackupCodes = () => const fetchBackupCodes = () =>
(dispatch: AppDispatch, getState: () => RootState) => { (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) => { 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; return data;
}).catch((error: unknown) => { }).catch((error: unknown) => {
dispatch({ type: MFA_BACKUP_CODES_FETCH_FAIL }); dispatch<MfaAction>({ type: MFA_BACKUP_CODES_FETCH_FAIL });
throw error; throw error;
}); });
}; };
const setupMfa = (method: 'totp') => const setupMfa = (method: 'totp') =>
(dispatch: AppDispatch, getState: () => RootState) => { (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) => { return getClient(getState).settings.mfa.getMfaSetup(method).then((data) => {
dispatch({ type: MFA_SETUP_SUCCESS, data }); dispatch<MfaAction>({ type: MFA_SETUP_SUCCESS, data });
return data; return data;
}).catch((error: unknown) => { }).catch((error: unknown) => {
dispatch({ type: MFA_SETUP_FAIL }); dispatch<MfaAction>({ type: MFA_SETUP_FAIL });
throw error; throw error;
}); });
}; };
const confirmMfa = (method: 'totp', code: string, password: string) => const confirmMfa = (method: 'totp', code: string, password: string) =>
(dispatch: AppDispatch, getState: () => RootState) => { (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) => { 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; return data;
}).catch((error: unknown) => { }).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; throw error;
}); });
}; };
const disableMfa = (method: 'totp', password: string) => const disableMfa = (method: 'totp', password: string) =>
(dispatch: AppDispatch, getState: () => RootState) => { (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) => { 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; return data;
}).catch((error: unknown) => { }).catch((error: unknown) => {
dispatch({ type: MFA_DISABLE_FAIL, method, skipAlert: true }); dispatch<MfaAction>({ type: MFA_DISABLE_FAIL, method, skipAlert: true });
throw error; 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 { export {
MFA_FETCH_REQUEST, MFA_FETCH_REQUEST,
MFA_FETCH_SUCCESS, MFA_FETCH_SUCCESS,
@ -101,4 +119,5 @@ export {
setupMfa, setupMfa,
confirmMfa, confirmMfa,
disableMfa, disableMfa,
type MfaAction,
}; };

View file

@ -4,7 +4,7 @@ import { getClient } from '../api';
import { importEntities } from './importer'; 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'; import type { AppDispatch, RootState } from 'pl-fe/store';
const PINNED_STATUSES_FETCH_REQUEST = 'PINNED_STATUSES_FETCH_REQUEST' as const; 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 => { return getClient(getState()).accounts.getAccountStatuses(me as string, { pinned: true }).then(response => {
dispatch(importEntities({ statuses: response.items })); dispatch(importEntities({ statuses: response.items }));
dispatch(fetchPinnedStatusesSuccess(response.items, null)); dispatch(fetchPinnedStatusesSuccess(response.items, response.next));
}).catch(error => { }).catch(error => {
dispatch(fetchPinnedStatusesFail(error)); dispatch(fetchPinnedStatusesFail(error));
}); });
@ -30,7 +30,7 @@ const fetchPinnedStatusesRequest = () => ({
type: PINNED_STATUSES_FETCH_REQUEST, 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, type: PINNED_STATUSES_FETCH_SUCCESS,
statuses, statuses,
next, next,

View file

@ -11,6 +11,8 @@ import { normalizeUsername } from 'pl-fe/utils/input';
import { AUTH_LOGGED_OUT, messages } from './auth'; 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'; import type { AppDispatch, RootState } from 'pl-fe/store';
const FETCH_TOKENS_REQUEST = 'FETCH_TOKENS_REQUEST' as const; const FETCH_TOKENS_REQUEST = 'FETCH_TOKENS_REQUEST' as const;
@ -47,32 +49,32 @@ const MOVE_ACCOUNT_FAIL = 'MOVE_ACCOUNT_FAIL' as const;
const fetchOAuthTokens = () => const fetchOAuthTokens = () =>
(dispatch: AppDispatch, getState: () => RootState) => { (dispatch: AppDispatch, getState: () => RootState) => {
dispatch({ type: FETCH_TOKENS_REQUEST }); dispatch<SecurityAction>({ type: FETCH_TOKENS_REQUEST });
return getClient(getState).settings.getOauthTokens().then((tokens) => { return getClient(getState).settings.getOauthTokens().then((tokens) => {
dispatch({ type: FETCH_TOKENS_SUCCESS, tokens }); dispatch<SecurityAction>({ type: FETCH_TOKENS_SUCCESS, tokens });
}).catch((e) => { }).catch((e) => {
dispatch({ type: FETCH_TOKENS_FAIL }); dispatch<SecurityAction>({ type: FETCH_TOKENS_FAIL });
}); });
}; };
const revokeOAuthTokenById = (tokenId: number) => const revokeOAuthTokenById = (tokenId: number) =>
(dispatch: AppDispatch, getState: () => RootState) => { (dispatch: AppDispatch, getState: () => RootState) => {
dispatch({ type: REVOKE_TOKEN_REQUEST, tokenId }); dispatch<SecurityAction>({ type: REVOKE_TOKEN_REQUEST, tokenId });
return getClient(getState).settings.deleteOauthToken(tokenId).then(() => { return getClient(getState).settings.deleteOauthToken(tokenId).then(() => {
dispatch({ type: REVOKE_TOKEN_SUCCESS, tokenId }); dispatch<SecurityAction>({ type: REVOKE_TOKEN_SUCCESS, tokenId });
}).catch(() => { }).catch(() => {
dispatch({ type: REVOKE_TOKEN_FAIL, tokenId }); dispatch<SecurityAction>({ type: REVOKE_TOKEN_FAIL, tokenId });
}); });
}; };
const changePassword = (oldPassword: string, newPassword: string) => const changePassword = (oldPassword: string, newPassword: string) =>
(dispatch: AppDispatch, getState: () => RootState) => { (dispatch: AppDispatch, getState: () => RootState) => {
dispatch({ type: CHANGE_PASSWORD_REQUEST }); dispatch<SecurityAction>({ type: CHANGE_PASSWORD_REQUEST });
return getClient(getState).settings.changePassword(oldPassword, newPassword).then(response => { return getClient(getState).settings.changePassword(oldPassword, newPassword).then(response => {
dispatch({ type: CHANGE_PASSWORD_SUCCESS, response }); dispatch<SecurityAction>({ type: CHANGE_PASSWORD_SUCCESS, response });
}).catch(error => { }).catch(error => {
dispatch({ type: CHANGE_PASSWORD_FAIL, error, skipAlert: true }); dispatch<SecurityAction>({ type: CHANGE_PASSWORD_FAIL, error, skipAlert: true });
throw error; throw error;
}); });
}; };
@ -81,58 +83,83 @@ const resetPassword = (usernameOrEmail: string) =>
(dispatch: AppDispatch, getState: () => RootState) => { (dispatch: AppDispatch, getState: () => RootState) => {
const input = normalizeUsername(usernameOrEmail); const input = normalizeUsername(usernameOrEmail);
dispatch({ type: RESET_PASSWORD_REQUEST }); dispatch<SecurityAction>({ type: RESET_PASSWORD_REQUEST });
return getClient(getState).settings.resetPassword( return getClient(getState).settings.resetPassword(
input.includes('@') ? input : undefined, input.includes('@') ? input : undefined,
input.includes('@') ? undefined : input, input.includes('@') ? undefined : input,
).then(() => { ).then(() => {
dispatch({ type: RESET_PASSWORD_SUCCESS }); dispatch<SecurityAction>({ type: RESET_PASSWORD_SUCCESS });
}).catch(error => { }).catch(error => {
dispatch({ type: RESET_PASSWORD_FAIL, error }); dispatch<SecurityAction>({ type: RESET_PASSWORD_FAIL, error });
throw error; throw error;
}); });
}; };
const changeEmail = (email: string, password: string) => const changeEmail = (email: string, password: string) =>
(dispatch: AppDispatch, getState: () => RootState) => { (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 => { 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 => { }).catch(error => {
dispatch({ type: CHANGE_EMAIL_FAIL, email, error, skipAlert: true }); dispatch<SecurityAction>({ type: CHANGE_EMAIL_FAIL, email, error, skipAlert: true });
throw error; throw error;
}); });
}; };
const deleteAccount = (password: string) => const deleteAccount = (password: string) =>
(dispatch: AppDispatch, getState: () => RootState) => { (dispatch: AppDispatch, getState: () => RootState) => {
dispatch({ type: CHANGE_PASSWORD_REQUEST }); dispatch<SecurityAction>({ type: CHANGE_PASSWORD_REQUEST });
const account = getLoggedInAccount(getState()); const account = getLoggedInAccount(getState())!;
dispatch({ type: DELETE_ACCOUNT_REQUEST }); dispatch<SecurityAction>({ type: DELETE_ACCOUNT_REQUEST });
return getClient(getState).settings.deleteAccount(password).then(response => { return getClient(getState).settings.deleteAccount(password).then(response => {
dispatch({ type: DELETE_ACCOUNT_SUCCESS, response }); dispatch<SecurityAction>({ type: DELETE_ACCOUNT_SUCCESS, response });
dispatch({ type: AUTH_LOGGED_OUT, account }); dispatch<SecurityAction>({ type: AUTH_LOGGED_OUT, account });
toast.success(messages.loggedOut); toast.success(messages.loggedOut);
}).catch(error => { }).catch(error => {
dispatch({ type: DELETE_ACCOUNT_FAIL, error, skipAlert: true }); dispatch<SecurityAction>({ type: DELETE_ACCOUNT_FAIL, error, skipAlert: true });
throw error; throw error;
}); });
}; };
const moveAccount = (targetAccount: string, password: string) => const moveAccount = (targetAccount: string, password: string) =>
(dispatch: AppDispatch, getState: () => RootState) => { (dispatch: AppDispatch, getState: () => RootState) => {
dispatch({ type: MOVE_ACCOUNT_REQUEST }); dispatch<SecurityAction>({ type: MOVE_ACCOUNT_REQUEST });
return getClient(getState).settings.moveAccount(targetAccount, password).then(response => { return getClient(getState).settings.moveAccount(targetAccount, password).then(response => {
dispatch({ type: MOVE_ACCOUNT_SUCCESS, response }); dispatch<SecurityAction>({ type: MOVE_ACCOUNT_SUCCESS, response });
}).catch(error => { }).catch(error => {
dispatch({ type: MOVE_ACCOUNT_FAIL, error, skipAlert: true }); dispatch<SecurityAction>({ type: MOVE_ACCOUNT_FAIL, error, skipAlert: true });
throw error; 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 { export {
FETCH_TOKENS_REQUEST, FETCH_TOKENS_REQUEST,
FETCH_TOKENS_SUCCESS, FETCH_TOKENS_SUCCESS,
@ -165,4 +192,5 @@ export {
changeEmail, changeEmail,
deleteAccount, deleteAccount,
moveAccount, moveAccount,
type SecurityAction,
}; };

View file

@ -23,7 +23,8 @@ import { ME_FETCH_SKIP, type MeAction } from '../actions/me';
import type { PlfeResponse } from 'pl-fe/api'; import type { PlfeResponse } from 'pl-fe/api';
import type { Account as AccountEntity } from 'pl-fe/normalizers/account'; 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 : ''); 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 }; return { ...newState, client: newClient };
}; };
const reducer = (state: State, action: AnyAction | AuthAction | MeAction | PreloadAction): State => { const reducer = (state: State, action: Action): State => {
switch (action.type) { switch (action.type) {
case AUTH_APP_CREATED: case AUTH_APP_CREATED:
return updateState(state, (draft) => { return updateState(state, (draft) => {
@ -304,7 +305,7 @@ const reducer = (state: State, action: AnyAction | AuthAction | MeAction | Prelo
}); });
case AUTH_APP_AUTHORIZED: case AUTH_APP_AUTHORIZED:
return updateState(state, (draft) => { return updateState(state, (draft) => {
draft.app = ({ ...draft.app, ...action.token }); if (draft.app) draft.app = ({ ...draft.app, ...action.token });
}); });
case AUTH_LOGGED_IN: case AUTH_LOGGED_IN:
return updateState(state, (draft) => { return updateState(state, (draft) => {
@ -328,17 +329,20 @@ const reducer = (state: State, action: AnyAction | AuthAction | MeAction | Prelo
}); });
case VERIFY_CREDENTIALS_FAIL: case VERIFY_CREDENTIALS_FAIL:
return updateState(state, (draft) => { return updateState(state, (draft) => {
deleteForbiddenToken(draft, action.error, action.token); deleteForbiddenToken(draft, action.error as any, action.token);
}); });
case SWITCH_ACCOUNT: case SWITCH_ACCOUNT:
return updateState(state, (draft) => { return updateState(state, (draft) => {
draft.me = action.account.url; draft.me = action.account.url;
}, () => { }, () => {
const accessToken = state.users[action.account.url]?.access_token;
if (state.client.baseURL === parseBaseURL(action.account.url)) { if (state.client.baseURL === parseBaseURL(action.account.url)) {
state.client.accessToken = action.account.access_token; state.client.accessToken = accessToken;
return state.client; 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: case ME_FETCH_SKIP:
return updateState(state, (draft) => { return updateState(state, (draft) => {
@ -373,7 +377,7 @@ const userSwitched = (oldState: State, state: State) => {
return stillValid && didChange && !userUpgradedUrl; 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 loggedOutStandalone = action.type === AUTH_LOGGED_OUT && action.standalone;
const switched = userSwitched(oldState, state); 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); const state = reducer(oldState, action);
if (state !== oldState) { if (state !== oldState) {
// Persist the state in localStorage // Persist the state in localStorage
persistAuth(state); 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 // Persist the session
persistSession(state); persistSession(state);

View file

@ -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) { if (error.response?.status === 401) {
return handleAuthFetch(state); return handleAuthFetch(state);
} else { } else {

View file

@ -4,11 +4,11 @@ import {
MFA_FETCH_SUCCESS, MFA_FETCH_SUCCESS,
MFA_CONFIRM_SUCCESS, MFA_CONFIRM_SUCCESS,
MFA_DISABLE_SUCCESS, MFA_DISABLE_SUCCESS,
type MfaAction,
} from '../actions/mfa'; } 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 { OauthToken } from 'pl-api';
import type { AnyAction } from 'redux';
interface State { interface State {
tokens: Array<OauthToken>; 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 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) { switch (action.type) {
case FETCH_TOKENS_SUCCESS: case FETCH_TOKENS_SUCCESS:
return create(state, (draft) => { return create(state, (draft) => {

View file

@ -1,16 +1,5 @@
import { create } from 'mutative'; 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 { import {
BOOKMARKED_STATUSES_FETCH_REQUEST, BOOKMARKED_STATUSES_FETCH_REQUEST,
BOOKMARKED_STATUSES_FETCH_SUCCESS, BOOKMARKED_STATUSES_FETCH_SUCCESS,
@ -19,7 +8,7 @@ import {
BOOKMARKED_STATUSES_EXPAND_SUCCESS, BOOKMARKED_STATUSES_EXPAND_SUCCESS,
BOOKMARKED_STATUSES_EXPAND_FAIL, BOOKMARKED_STATUSES_EXPAND_FAIL,
type BookmarksAction, type BookmarksAction,
} from '../actions/bookmarks'; } from 'pl-fe/actions/bookmarks';
import { import {
RECENT_EVENTS_FETCH_REQUEST, RECENT_EVENTS_FETCH_REQUEST,
RECENT_EVENTS_FETCH_SUCCESS, RECENT_EVENTS_FETCH_SUCCESS,
@ -28,7 +17,7 @@ import {
JOINED_EVENTS_FETCH_SUCCESS, JOINED_EVENTS_FETCH_SUCCESS,
JOINED_EVENTS_FETCH_FAIL, JOINED_EVENTS_FETCH_FAIL,
type EventsAction, type EventsAction,
} from '../actions/events'; } from 'pl-fe/actions/events';
import { import {
FAVOURITED_STATUSES_FETCH_REQUEST, FAVOURITED_STATUSES_FETCH_REQUEST,
FAVOURITED_STATUSES_FETCH_SUCCESS, FAVOURITED_STATUSES_FETCH_SUCCESS,
@ -43,7 +32,7 @@ import {
ACCOUNT_FAVOURITED_STATUSES_EXPAND_SUCCESS, ACCOUNT_FAVOURITED_STATUSES_EXPAND_SUCCESS,
ACCOUNT_FAVOURITED_STATUSES_EXPAND_FAIL, ACCOUNT_FAVOURITED_STATUSES_EXPAND_FAIL,
type FavouritesAction, type FavouritesAction,
} from '../actions/favourites'; } from 'pl-fe/actions/favourites';
import { import {
FAVOURITE_SUCCESS, FAVOURITE_SUCCESS,
UNFAVOURITE_SUCCESS, UNFAVOURITE_SUCCESS,
@ -52,8 +41,8 @@ import {
PIN_SUCCESS, PIN_SUCCESS,
UNPIN_SUCCESS, UNPIN_SUCCESS,
type InteractionsAction, type InteractionsAction,
} from '../actions/interactions'; } from 'pl-fe/actions/interactions';
import { PINNED_STATUSES_FETCH_SUCCESS, type PinStatusesAction } from '../actions/pin-statuses'; import { PINNED_STATUSES_FETCH_SUCCESS, type PinStatusesAction } from 'pl-fe/actions/pin-statuses';
import { import {
SCHEDULED_STATUSES_FETCH_REQUEST, SCHEDULED_STATUSES_FETCH_REQUEST,
SCHEDULED_STATUSES_FETCH_SUCCESS, SCHEDULED_STATUSES_FETCH_SUCCESS,
@ -63,10 +52,20 @@ import {
SCHEDULED_STATUSES_EXPAND_FAIL, SCHEDULED_STATUSES_EXPAND_FAIL,
SCHEDULED_STATUS_CANCEL_REQUEST, SCHEDULED_STATUS_CANCEL_REQUEST,
SCHEDULED_STATUS_CANCEL_SUCCESS, 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 { PaginatedResponse, ScheduledStatus, Status } from 'pl-api';
import type { AnyAction } from 'redux';
interface StatusList { interface StatusList {
next: (() => Promise<PaginatedResponse<Status>>) | null; 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) { switch (action.type) {
case FAVOURITED_STATUSES_FETCH_REQUEST: case FAVOURITED_STATUSES_FETCH_REQUEST:
case FAVOURITED_STATUSES_EXPAND_REQUEST: case FAVOURITED_STATUSES_EXPAND_REQUEST:
@ -209,9 +208,9 @@ const statusLists = (state = initialState, action: AnyAction | BookmarksAction |
case SCHEDULED_STATUSES_EXPAND_FAIL: case SCHEDULED_STATUSES_EXPAND_FAIL:
return create(state, draft => setLoading(draft, 'scheduled_statuses', false)); return create(state, draft => setLoading(draft, 'scheduled_statuses', false));
case SCHEDULED_STATUSES_FETCH_SUCCESS: 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: 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_REQUEST:
case SCHEDULED_STATUS_CANCEL_SUCCESS: case SCHEDULED_STATUS_CANCEL_SUCCESS:
return create(state, draft => removeOneFromList(draft, 'scheduled_statuses', action.statusId)); return create(state, draft => removeOneFromList(draft, 'scheduled_statuses', action.statusId));