refactor: user admin service (#23785)

This commit is contained in:
Jason Rasmussen
2025-11-11 07:42:33 -05:00
committed by GitHub
parent 2611e2ec20
commit 2f40f5aad8
15 changed files with 395 additions and 333 deletions

View File

@@ -1,11 +1,9 @@
<script lang="ts">
import { handleCreateUserAdmin } from '$lib/services/user-admin.service';
import { featureFlags } from '$lib/stores/server-config.store';
import { userInteraction } from '$lib/stores/user.svelte';
import { ByteUnit, convertToBytes } from '$lib/utils/byte-units';
import { handleError } from '$lib/utils/handle-error';
import { createUserAdmin, type UserAdminResponseDto } from '@immich/sdk';
import {
Alert,
Button,
Field,
HelperText,
@@ -20,13 +18,12 @@
} from '@immich/ui';
import { t } from 'svelte-i18n';
interface Props {
onClose: (user?: UserAdminResponseDto) => void;
}
type Props = {
onClose: () => void;
};
let { onClose }: Props = $props();
let error = $state('');
let success = $state(false);
let email = $state('');
@@ -57,40 +54,28 @@
}
isCreatingUser = true;
error = '';
try {
const user = await createUserAdmin({
userAdminCreateDto: {
email,
password,
shouldChangePassword,
name,
quotaSizeInBytes,
notify,
isAdmin,
},
});
const success = await handleCreateUserAdmin({
email,
password,
shouldChangePassword,
name,
quotaSizeInBytes,
notify,
isAdmin,
});
success = true;
onClose(user);
return;
} catch (error) {
handleError(error, $t('errors.unable_to_create_user'));
} finally {
isCreatingUser = false;
if (success) {
onClose();
}
isCreatingUser = false;
};
</script>
<Modal title={$t('create_new_user')} {onClose} size="small">
<ModalBody>
<form onsubmit={onSubmit} autocomplete="off" id="create-new-user-form">
{#if error}
<Alert color="danger" size="small" title={error} closable />
{/if}
{#if success}
<p class="text-sm text-immich-primary">{$t('new_user_created')}</p>
{/if}

View File

@@ -1,14 +1,15 @@
<script lang="ts">
import FormatMessage from '$lib/elements/FormatMessage.svelte';
import { handleDeleteUserAdmin } from '$lib/services/user-admin.service';
import { serverConfig } from '$lib/stores/server-config.store';
import { handleError } from '$lib/utils/handle-error';
import { deleteUserAdmin, type UserAdminResponseDto, type UserResponseDto } from '@immich/sdk';
import { type UserAdminResponseDto } from '@immich/sdk';
import { Alert, Checkbox, ConfirmModal, Field, Input, Label, Text } from '@immich/ui';
import { mdiTrashCanOutline } from '@mdi/js';
import { t } from 'svelte-i18n';
type Props = {
user: UserResponseDto;
onClose: (user?: UserAdminResponseDto) => void;
user: UserAdminResponseDto;
onClose: () => void;
};
let { user, onClose }: Props = $props();
@@ -17,22 +18,21 @@
let email = $state('');
let disabled = $derived(force && email !== user.email);
const handleClose = async (confirmed: boolean) => {
const handleClose = async (confirmed?: boolean) => {
if (!confirmed) {
onClose();
return;
}
try {
const result = await deleteUserAdmin({ id: user.id, userAdminDeleteDto: { force } });
onClose(result);
} catch (error) {
handleError(error, $t('errors.unable_to_delete_user'));
const success = await handleDeleteUserAdmin(user, { force });
if (success) {
onClose();
}
};
</script>
<ConfirmModal
icon={mdiTrashCanOutline}
title={$t('delete_user')}
confirmText={force ? $t('permanently_delete') : $t('delete')}
onClose={handleClose}

View File

@@ -1,10 +1,10 @@
<script lang="ts">
import { AppRoute } from '$lib/constants';
import { handleUpdateUserAdmin } from '$lib/services/user-admin.service';
import { user as authUser } from '$lib/stores/user.store';
import { userInteraction } from '$lib/stores/user.svelte';
import { ByteUnit, convertFromBytes, convertToBytes } from '$lib/utils/byte-units';
import { handleError } from '$lib/utils/handle-error';
import { updateUserAdmin, type UserAdminResponseDto } from '@immich/sdk';
import { type UserAdminResponseDto } from '@immich/sdk';
import {
Button,
Field,
@@ -23,7 +23,7 @@
interface Props {
user: UserAdminResponseDto;
onClose: (data?: UserAdminResponseDto) => void;
onClose: () => void;
}
let { user, onClose }: Props = $props();
@@ -48,28 +48,20 @@
quotaSizeBytes > userInteraction.serverInfo.diskSizeRaw,
);
const handleEditUser = async () => {
try {
const newUser = await updateUserAdmin({
id: user.id,
userAdminUpdateDto: {
email,
name,
storageLabel,
quotaSizeInBytes: typeof quotaSize === 'number' ? convertToBytes(quotaSize, ByteUnit.GiB) : null,
isAdmin,
},
});
onClose(newUser);
} catch (error) {
handleError(error, $t('errors.unable_to_update_user'));
}
};
const onSubmit = async (event: Event) => {
event.preventDefault();
await handleEditUser();
const success = await handleUpdateUserAdmin(user, {
email,
name,
storageLabel,
quotaSizeInBytes: typeof quotaSize === 'number' ? convertToBytes(quotaSize, ByteUnit.GiB) : null,
isAdmin,
});
if (success) {
onClose();
}
};
</script>

View File

@@ -1,30 +1,39 @@
<script lang="ts">
import FormatMessage from '$lib/elements/FormatMessage.svelte';
import { handleError } from '$lib/utils/handle-error';
import { restoreUserAdmin, type UserAdminResponseDto, type UserResponseDto } from '@immich/sdk';
import { Button, HStack, Modal, ModalBody, ModalFooter } from '@immich/ui';
import { handleRestoreUserAdmin } from '$lib/services/user-admin.service';
import { type UserAdminResponseDto } from '@immich/sdk';
import { ConfirmModal } from '@immich/ui';
import { mdiDeleteRestore } from '@mdi/js';
import { t } from 'svelte-i18n';
interface Props {
user: UserResponseDto;
onClose: (user?: UserAdminResponseDto) => void;
}
type Props = {
user: UserAdminResponseDto;
onClose: () => void;
};
let { user, onClose }: Props = $props();
const handleRestoreUser = async () => {
try {
const result = await restoreUserAdmin({ id: user.id });
onClose(result);
} catch (error) {
handleError(error, $t('errors.unable_to_restore_user'));
const handleClose = async (confirmed: boolean) => {
if (!confirmed) {
return;
}
const success = await handleRestoreUserAdmin(user);
if (success) {
onClose();
}
};
</script>
<Modal title={$t('restore_user')} {onClose} icon={mdiDeleteRestore} size="small">
<ModalBody>
<ConfirmModal
icon={mdiDeleteRestore}
title={$t('restore_user')}
confirmText={$t('restore')}
confirmColor="primary"
size="small"
onClose={handleClose}
>
{#snippet promptSnippet()}
<p>
<FormatMessage key="admin.user_restore_description" values={{ user: user.name }}>
{#snippet children({ message })}
@@ -32,16 +41,5 @@
{/snippet}
</FormatMessage>
</p>
</ModalBody>
<ModalFooter>
<HStack fullWidth>
<Button shape="round" color="secondary" fullWidth onclick={() => onClose()}>
{$t('cancel')}
</Button>
<Button shape="round" color="primary" fullWidth onclick={() => handleRestoreUser()}>
{$t('restore')}
</Button>
</HStack>
</ModalFooter>
</Modal>
{/snippet}
</ConfirmModal>