Compare commits
3 Commits
main
...
07ae9b61fb
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
07ae9b61fb | ||
|
|
1482cb7a06 | ||
|
|
9d77685721 |
@@ -1,7 +1,8 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { enhance } from '$app/forms';
|
import { enhance } from '$app/forms';
|
||||||
import { beforeNavigate, goto } from '$app/navigation';
|
|
||||||
import { m } from '$lib/paraglide/messages.js';
|
import { m } from '$lib/paraglide/messages.js';
|
||||||
|
import { createUnsavedWarning } from '$lib/shared/hooks/useUnsavedWarning.svelte';
|
||||||
|
import UnsavedWarningBanner from '$lib/shared/primitives/UnsavedWarningBanner.svelte';
|
||||||
|
|
||||||
const availableStandard = $derived([
|
const availableStandard = $derived([
|
||||||
{ value: 'READ_ALL', label: m.admin_perm_read_all() },
|
{ value: 'READ_ALL', label: m.admin_perm_read_all() },
|
||||||
@@ -18,17 +19,7 @@ const availableAdmin = $derived([
|
|||||||
|
|
||||||
let { form } = $props();
|
let { form } = $props();
|
||||||
|
|
||||||
let isDirty = $state(false);
|
const unsaved = createUnsavedWarning();
|
||||||
let showUnsavedWarning = $state(false);
|
|
||||||
let discardTarget: string | null = $state(null);
|
|
||||||
|
|
||||||
beforeNavigate(({ cancel, to }) => {
|
|
||||||
if (isDirty) {
|
|
||||||
cancel();
|
|
||||||
showUnsavedWarning = true;
|
|
||||||
discardTarget = to?.url.href ?? null;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="flex flex-1 flex-col overflow-hidden">
|
<div class="flex flex-1 flex-col overflow-hidden">
|
||||||
@@ -58,23 +49,8 @@ beforeNavigate(({ cancel, to }) => {
|
|||||||
|
|
||||||
<!-- Scrollable body -->
|
<!-- Scrollable body -->
|
||||||
<div class="flex-1 overflow-y-auto px-5 py-5">
|
<div class="flex-1 overflow-y-auto px-5 py-5">
|
||||||
{#if showUnsavedWarning}
|
{#if unsaved.showUnsavedWarning}
|
||||||
<div
|
<UnsavedWarningBanner onDiscard={unsaved.discard} />
|
||||||
class="mb-5 flex items-center justify-between rounded border border-amber-200 bg-amber-50 p-3 text-sm text-amber-800 dark:border-amber-800 dark:bg-amber-950/40 dark:text-amber-300"
|
|
||||||
>
|
|
||||||
<span>{m.admin_unsaved_warning()}</span>
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
onclick={() => {
|
|
||||||
isDirty = false;
|
|
||||||
showUnsavedWarning = false;
|
|
||||||
if (discardTarget) goto(discardTarget);
|
|
||||||
}}
|
|
||||||
class="ml-4 shrink-0 font-sans text-xs font-bold tracking-widest text-amber-800 uppercase hover:text-amber-900 dark:text-amber-300"
|
|
||||||
>
|
|
||||||
{m.person_discard_changes()}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
{/if}
|
{/if}
|
||||||
{#if form?.error}
|
{#if form?.error}
|
||||||
<div class="mb-5 rounded border border-red-200 bg-red-50 p-3 text-sm text-red-700">
|
<div class="mb-5 rounded border border-red-200 bg-red-50 p-3 text-sm text-red-700">
|
||||||
@@ -85,11 +61,11 @@ beforeNavigate(({ cancel, to }) => {
|
|||||||
<form
|
<form
|
||||||
id="new-group-form"
|
id="new-group-form"
|
||||||
method="POST"
|
method="POST"
|
||||||
use:enhance
|
use:enhance={() => async ({ result, update }) => {
|
||||||
oninput={() => {
|
if (result.type === 'redirect') unsaved.clearOnSuccess();
|
||||||
isDirty = true;
|
await update();
|
||||||
showUnsavedWarning = false;
|
|
||||||
}}
|
}}
|
||||||
|
oninput={unsaved.markDirty}
|
||||||
class="space-y-5"
|
class="space-y-5"
|
||||||
>
|
>
|
||||||
<!-- Name card -->
|
<!-- Name card -->
|
||||||
|
|||||||
130
frontend/src/routes/admin/groups/new/page.svelte.spec.ts
Normal file
130
frontend/src/routes/admin/groups/new/page.svelte.spec.ts
Normal file
@@ -0,0 +1,130 @@
|
|||||||
|
import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
|
||||||
|
import { cleanup, render } from 'vitest-browser-svelte';
|
||||||
|
import { page } from 'vitest/browser';
|
||||||
|
import Page from './+page.svelte';
|
||||||
|
|
||||||
|
const enhanceCaptureRef = vi.hoisted(() => ({ submitFn: undefined as unknown }));
|
||||||
|
|
||||||
|
vi.mock('$app/forms', () => ({
|
||||||
|
enhance: (_el: HTMLFormElement, fn?: unknown) => {
|
||||||
|
enhanceCaptureRef.submitFn = fn;
|
||||||
|
return { destroy: vi.fn() };
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
vi.mock('$app/navigation', () => ({ beforeNavigate: vi.fn(), goto: vi.fn() }));
|
||||||
|
|
||||||
|
import { beforeNavigate, goto } from '$app/navigation';
|
||||||
|
|
||||||
|
afterEach(cleanup);
|
||||||
|
|
||||||
|
// ─── Unsaved-changes guard ────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
describe('Admin new group page – unsaved-changes guard', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
vi.clearAllMocks();
|
||||||
|
enhanceCaptureRef.submitFn = undefined;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not show unsaved warning initially', async () => {
|
||||||
|
render(Page, { props: { form: null } });
|
||||||
|
await expect.element(page.getByText(/ungespeicherte Änderungen/i)).not.toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('cancels navigation and shows banner when form is dirty', async () => {
|
||||||
|
render(Page, { props: { form: null } });
|
||||||
|
const [callback] = vi.mocked(beforeNavigate).mock.calls[0];
|
||||||
|
|
||||||
|
document
|
||||||
|
.querySelector<HTMLInputElement>('input[name="name"]')!
|
||||||
|
.dispatchEvent(new InputEvent('input', { bubbles: true }));
|
||||||
|
|
||||||
|
const cancel = vi.fn();
|
||||||
|
callback({ cancel, to: { url: new URL('http://localhost/admin/groups') } });
|
||||||
|
|
||||||
|
expect(cancel).toHaveBeenCalled();
|
||||||
|
await expect.element(page.getByText(/ungespeicherte Änderungen/i)).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not cancel navigation when form is clean', async () => {
|
||||||
|
render(Page, { props: { form: null } });
|
||||||
|
const [callback] = vi.mocked(beforeNavigate).mock.calls[0];
|
||||||
|
|
||||||
|
const cancel = vi.fn();
|
||||||
|
callback({ cancel, to: { url: new URL('http://localhost/admin/groups') } });
|
||||||
|
|
||||||
|
expect(cancel).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('discard button calls goto with the target URL', async () => {
|
||||||
|
render(Page, { props: { form: null } });
|
||||||
|
const [callback] = vi.mocked(beforeNavigate).mock.calls[0];
|
||||||
|
|
||||||
|
document
|
||||||
|
.querySelector<HTMLInputElement>('input[name="name"]')!
|
||||||
|
.dispatchEvent(new InputEvent('input', { bubbles: true }));
|
||||||
|
|
||||||
|
callback({ cancel: vi.fn(), to: { url: new URL('http://localhost/admin/groups') } });
|
||||||
|
|
||||||
|
await page.getByRole('button', { name: /verwerfen/i }).click();
|
||||||
|
|
||||||
|
expect(vi.mocked(goto)).toHaveBeenCalledWith('http://localhost/admin/groups');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('clears banner when enhance callback receives a redirect result', async () => {
|
||||||
|
render(Page, { props: { form: null } });
|
||||||
|
const [navCallback] = vi.mocked(beforeNavigate).mock.calls[0];
|
||||||
|
|
||||||
|
document
|
||||||
|
.querySelector<HTMLInputElement>('input[name="name"]')!
|
||||||
|
.dispatchEvent(new InputEvent('input', { bubbles: true }));
|
||||||
|
|
||||||
|
navCallback({ cancel: vi.fn(), to: { url: new URL('http://localhost/admin/groups') } });
|
||||||
|
await expect.element(page.getByText(/ungespeicherte Änderungen/i)).toBeInTheDocument();
|
||||||
|
|
||||||
|
type SubmitFn = () => Promise<
|
||||||
|
(opts: {
|
||||||
|
result: { type: string; [key: string]: unknown };
|
||||||
|
update: () => Promise<void>;
|
||||||
|
}) => Promise<void>
|
||||||
|
>;
|
||||||
|
const innerFn = await (enhanceCaptureRef.submitFn as SubmitFn)();
|
||||||
|
await innerFn({
|
||||||
|
result: { type: 'redirect', location: '/admin/groups', status: 303 },
|
||||||
|
update: vi.fn().mockResolvedValue(undefined)
|
||||||
|
});
|
||||||
|
|
||||||
|
await expect.element(page.getByText(/ungespeicherte Änderungen/i)).not.toBeInTheDocument();
|
||||||
|
|
||||||
|
const cancel = vi.fn();
|
||||||
|
navCallback({ cancel, to: { url: new URL('http://localhost/admin/groups') } });
|
||||||
|
expect(cancel).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('keeps banner when enhance callback receives a failure result', async () => {
|
||||||
|
render(Page, { props: { form: null } });
|
||||||
|
const [navCallback] = vi.mocked(beforeNavigate).mock.calls[0];
|
||||||
|
|
||||||
|
document
|
||||||
|
.querySelector<HTMLInputElement>('input[name="name"]')!
|
||||||
|
.dispatchEvent(new InputEvent('input', { bubbles: true }));
|
||||||
|
|
||||||
|
navCallback({ cancel: vi.fn(), to: { url: new URL('http://localhost/admin/groups') } });
|
||||||
|
await expect.element(page.getByText(/ungespeicherte Änderungen/i)).toBeInTheDocument();
|
||||||
|
|
||||||
|
type SubmitFn = () => Promise<
|
||||||
|
(opts: {
|
||||||
|
result: { type: string; [key: string]: unknown };
|
||||||
|
update: () => Promise<void>;
|
||||||
|
}) => Promise<void>
|
||||||
|
>;
|
||||||
|
const innerFn = await (enhanceCaptureRef.submitFn as SubmitFn)();
|
||||||
|
await innerFn({
|
||||||
|
result: { type: 'failure', status: 400, data: { error: 'Name bereits vergeben' } },
|
||||||
|
update: vi.fn().mockResolvedValue(undefined)
|
||||||
|
});
|
||||||
|
|
||||||
|
const cancel = vi.fn();
|
||||||
|
navCallback({ cancel, to: { url: new URL('http://localhost/admin/groups') } });
|
||||||
|
expect(cancel).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -1,24 +1,15 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { enhance } from '$app/forms';
|
import { enhance } from '$app/forms';
|
||||||
import { beforeNavigate, goto } from '$app/navigation';
|
|
||||||
import { m } from '$lib/paraglide/messages.js';
|
import { m } from '$lib/paraglide/messages.js';
|
||||||
import UserProfileSection from '$lib/user/UserProfileSection.svelte';
|
import UserProfileSection from '$lib/user/UserProfileSection.svelte';
|
||||||
import UserGroupsSection from '$lib/user/UserGroupsSection.svelte';
|
import UserGroupsSection from '$lib/user/UserGroupsSection.svelte';
|
||||||
import AccountSection from './AccountSection.svelte';
|
import AccountSection from './AccountSection.svelte';
|
||||||
|
import { createUnsavedWarning } from '$lib/shared/hooks/useUnsavedWarning.svelte';
|
||||||
|
import UnsavedWarningBanner from '$lib/shared/primitives/UnsavedWarningBanner.svelte';
|
||||||
|
|
||||||
let { data, form } = $props();
|
let { data, form } = $props();
|
||||||
|
|
||||||
let isDirty = $state(false);
|
const unsaved = createUnsavedWarning();
|
||||||
let showUnsavedWarning = $state(false);
|
|
||||||
let discardTarget: string | null = $state(null);
|
|
||||||
|
|
||||||
beforeNavigate(({ cancel, to }) => {
|
|
||||||
if (isDirty) {
|
|
||||||
cancel();
|
|
||||||
showUnsavedWarning = true;
|
|
||||||
discardTarget = to?.url.href ?? null;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="flex flex-1 flex-col overflow-hidden">
|
<div class="flex flex-1 flex-col overflow-hidden">
|
||||||
@@ -44,23 +35,8 @@ beforeNavigate(({ cancel, to }) => {
|
|||||||
|
|
||||||
<!-- Scrollable body -->
|
<!-- Scrollable body -->
|
||||||
<div class="flex-1 overflow-y-auto px-5 py-5">
|
<div class="flex-1 overflow-y-auto px-5 py-5">
|
||||||
{#if showUnsavedWarning}
|
{#if unsaved.showUnsavedWarning}
|
||||||
<div
|
<UnsavedWarningBanner onDiscard={unsaved.discard} />
|
||||||
class="mb-5 flex items-center justify-between rounded border border-amber-200 bg-amber-50 p-3 text-sm text-amber-800 dark:border-amber-800 dark:bg-amber-950/40 dark:text-amber-300"
|
|
||||||
>
|
|
||||||
<span>{m.admin_unsaved_warning()}</span>
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
onclick={() => {
|
|
||||||
isDirty = false;
|
|
||||||
showUnsavedWarning = false;
|
|
||||||
if (discardTarget) goto(discardTarget);
|
|
||||||
}}
|
|
||||||
class="ml-4 shrink-0 font-sans text-xs font-bold tracking-widest text-amber-800 uppercase hover:text-amber-900 dark:text-amber-300"
|
|
||||||
>
|
|
||||||
{m.person_discard_changes()}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
{/if}
|
{/if}
|
||||||
{#if form?.error}
|
{#if form?.error}
|
||||||
<div class="mb-5 rounded border border-red-200 bg-red-50 p-3 text-sm text-red-700">
|
<div class="mb-5 rounded border border-red-200 bg-red-50 p-3 text-sm text-red-700">
|
||||||
@@ -71,11 +47,11 @@ beforeNavigate(({ cancel, to }) => {
|
|||||||
<form
|
<form
|
||||||
id="new-user-form"
|
id="new-user-form"
|
||||||
method="POST"
|
method="POST"
|
||||||
use:enhance
|
use:enhance={() => async ({ result, update }) => {
|
||||||
oninput={() => {
|
if (result.type === 'redirect') unsaved.clearOnSuccess();
|
||||||
isDirty = true;
|
await update();
|
||||||
showUnsavedWarning = false;
|
|
||||||
}}
|
}}
|
||||||
|
oninput={unsaved.markDirty}
|
||||||
class="space-y-5"
|
class="space-y-5"
|
||||||
>
|
>
|
||||||
<div class="rounded-sm border border-line bg-surface p-5 shadow-sm">
|
<div class="rounded-sm border border-line bg-surface p-5 shadow-sm">
|
||||||
|
|||||||
@@ -1,9 +1,19 @@
|
|||||||
import { afterEach, describe, expect, it, vi } from 'vitest';
|
import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
|
||||||
import { cleanup, render } from 'vitest-browser-svelte';
|
import { cleanup, render } from 'vitest-browser-svelte';
|
||||||
import { page } from 'vitest/browser';
|
import { page } from 'vitest/browser';
|
||||||
import Page from './+page.svelte';
|
import Page from './+page.svelte';
|
||||||
|
|
||||||
vi.mock('$app/forms', () => ({ enhance: () => () => {} }));
|
const enhanceCaptureRef = vi.hoisted(() => ({ submitFn: undefined as unknown }));
|
||||||
|
|
||||||
|
vi.mock('$app/forms', () => ({
|
||||||
|
enhance: (_el: HTMLFormElement, fn?: unknown) => {
|
||||||
|
enhanceCaptureRef.submitFn = fn;
|
||||||
|
return { destroy: vi.fn() };
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
vi.mock('$app/navigation', () => ({ beforeNavigate: vi.fn(), goto: vi.fn() }));
|
||||||
|
|
||||||
|
import { beforeNavigate, goto } from '$app/navigation';
|
||||||
|
|
||||||
const groups = [
|
const groups = [
|
||||||
{ id: 'g1', name: 'Editoren', permissions: ['WRITE_ALL'] },
|
{ id: 'g1', name: 'Editoren', permissions: ['WRITE_ALL'] },
|
||||||
@@ -66,3 +76,115 @@ describe('Admin new user page – error display', () => {
|
|||||||
await expect.element(page.getByText('Ein Fehler ist aufgetreten.')).not.toBeInTheDocument();
|
await expect.element(page.getByText('Ein Fehler ist aufgetreten.')).not.toBeInTheDocument();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// ─── Unsaved-changes guard ────────────────────────────────────────────────────
|
||||||
|
|
||||||
|
describe('Admin new user page – unsaved-changes guard', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
vi.clearAllMocks();
|
||||||
|
enhanceCaptureRef.submitFn = undefined;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not show unsaved warning initially', async () => {
|
||||||
|
render(Page, { data: baseData, form: null });
|
||||||
|
await expect.element(page.getByText(/ungespeicherte Änderungen/i)).not.toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('cancels navigation and shows banner when form is dirty', async () => {
|
||||||
|
render(Page, { data: baseData, form: null });
|
||||||
|
const [callback] = vi.mocked(beforeNavigate).mock.calls[0];
|
||||||
|
|
||||||
|
document
|
||||||
|
.querySelector<HTMLInputElement>('input[name="email"]')!
|
||||||
|
.dispatchEvent(new InputEvent('input', { bubbles: true }));
|
||||||
|
|
||||||
|
const cancel = vi.fn();
|
||||||
|
callback({ cancel, to: { url: new URL('http://localhost/admin/users') } });
|
||||||
|
|
||||||
|
expect(cancel).toHaveBeenCalled();
|
||||||
|
await expect.element(page.getByText(/ungespeicherte Änderungen/i)).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not cancel navigation when form is clean', async () => {
|
||||||
|
render(Page, { data: baseData, form: null });
|
||||||
|
const [callback] = vi.mocked(beforeNavigate).mock.calls[0];
|
||||||
|
|
||||||
|
const cancel = vi.fn();
|
||||||
|
callback({ cancel, to: { url: new URL('http://localhost/admin/users') } });
|
||||||
|
|
||||||
|
expect(cancel).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('discard button calls goto with the target URL', async () => {
|
||||||
|
render(Page, { data: baseData, form: null });
|
||||||
|
const [callback] = vi.mocked(beforeNavigate).mock.calls[0];
|
||||||
|
|
||||||
|
document
|
||||||
|
.querySelector<HTMLInputElement>('input[name="email"]')!
|
||||||
|
.dispatchEvent(new InputEvent('input', { bubbles: true }));
|
||||||
|
|
||||||
|
callback({ cancel: vi.fn(), to: { url: new URL('http://localhost/admin/users') } });
|
||||||
|
|
||||||
|
await page.getByRole('button', { name: /verwerfen/i }).click();
|
||||||
|
|
||||||
|
expect(vi.mocked(goto)).toHaveBeenCalledWith('http://localhost/admin/users');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('clears banner when enhance callback receives a redirect result', async () => {
|
||||||
|
render(Page, { data: baseData, form: null });
|
||||||
|
const [navCallback] = vi.mocked(beforeNavigate).mock.calls[0];
|
||||||
|
|
||||||
|
document
|
||||||
|
.querySelector<HTMLInputElement>('input[name="email"]')!
|
||||||
|
.dispatchEvent(new InputEvent('input', { bubbles: true }));
|
||||||
|
|
||||||
|
navCallback({ cancel: vi.fn(), to: { url: new URL('http://localhost/admin/users') } });
|
||||||
|
await expect.element(page.getByText(/ungespeicherte Änderungen/i)).toBeInTheDocument();
|
||||||
|
|
||||||
|
type SubmitFn = () => Promise<
|
||||||
|
(opts: {
|
||||||
|
result: { type: string; [key: string]: unknown };
|
||||||
|
update: () => Promise<void>;
|
||||||
|
}) => Promise<void>
|
||||||
|
>;
|
||||||
|
const innerFn = await (enhanceCaptureRef.submitFn as SubmitFn)();
|
||||||
|
await innerFn({
|
||||||
|
result: { type: 'redirect', location: '/admin/users', status: 303 },
|
||||||
|
update: vi.fn().mockResolvedValue(undefined)
|
||||||
|
});
|
||||||
|
|
||||||
|
await expect.element(page.getByText(/ungespeicherte Änderungen/i)).not.toBeInTheDocument();
|
||||||
|
|
||||||
|
const cancel = vi.fn();
|
||||||
|
navCallback({ cancel, to: { url: new URL('http://localhost/admin/users') } });
|
||||||
|
expect(cancel).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('keeps banner when enhance callback receives a failure result', async () => {
|
||||||
|
render(Page, { data: baseData, form: null });
|
||||||
|
const [navCallback] = vi.mocked(beforeNavigate).mock.calls[0];
|
||||||
|
|
||||||
|
document
|
||||||
|
.querySelector<HTMLInputElement>('input[name="email"]')!
|
||||||
|
.dispatchEvent(new InputEvent('input', { bubbles: true }));
|
||||||
|
|
||||||
|
navCallback({ cancel: vi.fn(), to: { url: new URL('http://localhost/admin/users') } });
|
||||||
|
await expect.element(page.getByText(/ungespeicherte Änderungen/i)).toBeInTheDocument();
|
||||||
|
|
||||||
|
type SubmitFn = () => Promise<
|
||||||
|
(opts: {
|
||||||
|
result: { type: string; [key: string]: unknown };
|
||||||
|
update: () => Promise<void>;
|
||||||
|
}) => Promise<void>
|
||||||
|
>;
|
||||||
|
const innerFn = await (enhanceCaptureRef.submitFn as SubmitFn)();
|
||||||
|
await innerFn({
|
||||||
|
result: { type: 'failure', status: 400, data: { error: 'E-Mail bereits vergeben' } },
|
||||||
|
update: vi.fn().mockResolvedValue(undefined)
|
||||||
|
});
|
||||||
|
|
||||||
|
const cancel = vi.fn();
|
||||||
|
navCallback({ cancel, to: { url: new URL('http://localhost/admin/users') } });
|
||||||
|
expect(cancel).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|||||||
Reference in New Issue
Block a user