Files
api/web/store/account.ts
2023-11-06 13:53:29 -08:00

253 lines
7.1 KiB
TypeScript

import { useMutation } from '@vue/apollo-composable';
import { logErrorMessages } from '@vue/apollo-util';
import { defineStore, createPinia, setActivePinia } from 'pinia';
import { CONNECT_SIGN_IN, CONNECT_SIGN_OUT } from './account.fragment';
import { useCallbackStore } from '~/store/callbackActions';
import { useErrorsStore } from '~/store/errors';
import { useServerStore } from '~/store/server';
import { useUnraidApiStore } from '~/store/unraidApi';
import { ACCOUNT_CALLBACK } from '~/helpers/urls';
import type { ExternalSignIn, ExternalSignOut } from '~/store/callback';
/**
* @see https://stackoverflow.com/questions/73476371/using-pinia-with-vue-js-web-components
* @see https://github.com/vuejs/pinia/discussions/1085
*/
setActivePinia(createPinia());
export interface ConnectSignInMutationPayload {
apiKey: string;
email: string;
preferred_username: string;
}
export const useAccountStore = defineStore('account', () => {
const callbackStore = useCallbackStore();
const errorsStore = useErrorsStore();
const serverStore = useServerStore();
const unraidApiStore = useUnraidApiStore();
// State
const accountAction = ref<ExternalSignIn | ExternalSignOut>();
const accountActionHide = ref<boolean>(false);
const accountActionStatus = ref<'failed' | 'ready' | 'success' | 'updating' | 'waiting'>('ready');
/**
* Handling sign in / out via graph api
*/
const unraidApiClient = computed(() => unraidApiStore.unraidApiClient);
const connectSignInPayload = ref<ConnectSignInMutationPayload | undefined>();
const setConnectSignInPayload = (payload: ConnectSignInMutationPayload | undefined) => {
connectSignInPayload.value = payload;
if (payload) {
accountActionStatus.value = 'waiting';
}
};
const queueConnectSignOut = ref<boolean>(false);
const setQueueConnectSignOut = (data: boolean) => {
queueConnectSignOut.value = data;
if (data) {
accountActionStatus.value = 'waiting';
}
};
watchEffect(() => {
if (unraidApiClient.value && connectSignInPayload.value) {
// connectSignInMutation();
setTimeout(() => {
connectSignInMutation();
}, 250);
}
if (unraidApiClient.value && queueConnectSignOut.value) {
// connectSignOutMutation();
setTimeout(() => {
connectSignOutMutation();
}, 250);
}
});
// Getters
const accountActionType = computed(() => accountAction.value?.type);
// Actions
const recover = () => {
callbackStore.send(
ACCOUNT_CALLBACK.toString(),
[{
server: {
...serverStore.serverAccountPayload,
},
type: 'recover',
}],
serverStore.inIframe ? 'newTab' : undefined,
);
};
const replace = () => {
callbackStore.send(
ACCOUNT_CALLBACK.toString(),
[{
server: {
...serverStore.serverAccountPayload,
},
type: 'replace',
}],
serverStore.inIframe ? 'newTab' : undefined,
);
};
const signIn = () => {
callbackStore.send(
ACCOUNT_CALLBACK.toString(),
[{
server: {
...serverStore.serverAccountPayload,
},
type: 'signIn',
}],
serverStore.inIframe ? 'newTab' : undefined,
);
};
const signOut = () => {
callbackStore.send(
ACCOUNT_CALLBACK.toString(),
[{
server: {
...serverStore.serverAccountPayload,
},
type: 'signOut',
}],
serverStore.inIframe ? 'newTab' : undefined,
);
};
const trialExtend = () => {
callbackStore.send(
ACCOUNT_CALLBACK.toString(),
[{
server: {
...serverStore.serverAccountPayload,
},
type: 'trialExtend',
}],
serverStore.inIframe ? 'newTab' : undefined,
);
};
const trialStart = () => {
callbackStore.send(
ACCOUNT_CALLBACK.toString(),
[{
server: {
...serverStore.serverAccountPayload,
},
type: 'trialStart',
}],
serverStore.inIframe ? 'newTab' : undefined,
);
};
const connectSignInMutation = async () => {
if (!connectSignInPayload.value ||
(connectSignInPayload.value && (!connectSignInPayload.value.apiKey || !connectSignInPayload.value.email || !connectSignInPayload.value.preferred_username))
) {
accountActionStatus.value = 'failed';
return console.error('[connectSignInMutation] incorrect payload', connectSignInPayload.value);
}
accountActionStatus.value = 'updating';
const { mutate: signInMutation, onDone, onError } = await useMutation(CONNECT_SIGN_IN, {
variables: {
input: {
apiKey: connectSignInPayload.value.apiKey,
userInfo: {
email: connectSignInPayload.value.email,
preferred_username: connectSignInPayload.value.preferred_username,
}
}
}
});
signInMutation();
onDone((res) => {
if (res.data?.connectSignIn) {
accountActionStatus.value = 'success';
setConnectSignInPayload(undefined); // reset
return;
}
accountActionStatus.value = 'failed';
errorsStore.setError({
heading: 'unraid-api failed to update Connect account configuration',
message: 'Sign In mutation unsuccessful',
level: 'error',
ref: 'connectSignInMutation',
type: 'account',
});
});
onError((error) => {
logErrorMessages(error);
accountActionStatus.value = 'failed';
errorsStore.setError({
heading: 'unraid-api failed to update Connect account configuration',
message: error.message,
level: 'error',
ref: 'connectSignInMutation',
type: 'account',
});
});
};
const connectSignOutMutation = async () => {
accountActionStatus.value = 'updating';
// @todo is this still needed here with the change to a mutation?
// if (!serverStore.registered && accountAction.value && !accountAction.value?.user) {
// accountActionHide.value = true;
// accountActionStatus.value = 'success';
// return;
// }
const { mutate: signOutMutation, onDone, onError } = await useMutation(CONNECT_SIGN_OUT);
signOutMutation();
onDone((res) => {
console.debug('[connectSignOutMutation]', res);
accountActionStatus.value = 'success';
setQueueConnectSignOut(false); // reset
});
onError((error) => {
logErrorMessages(error);
accountActionStatus.value = 'failed';
errorsStore.setError({
heading: 'Failed to update Connect account configuration',
message: error.message,
level: 'error',
ref: 'connectSignOutMutation',
type: 'account',
});
});
};
const setAccountAction = (action: ExternalSignIn|ExternalSignOut) => {
console.debug('[setAccountAction]', { action });
accountAction.value = action;
};
return {
// State
accountAction,
accountActionHide,
accountActionStatus,
// Getters
accountActionType,
// Actions
recover,
replace,
signIn,
signOut,
trialExtend,
trialStart,
setAccountAction,
setConnectSignInPayload,
setQueueConnectSignOut,
};
});