feat: add session management page - closes #1

This commit is contained in:
axel 2025-04-19 00:58:43 +02:00
parent db4eaef079
commit 1e32591387
2 changed files with 68 additions and 0 deletions

View File

@ -0,0 +1,21 @@
import { FORBIDDEN, SUCCESS } from '$lib/server/commonResponses';
import { deleteSession, getUserSessions } from '$lib/server/sessions';
import { fail, type Actions, type ServerLoad } from '@sveltejs/kit';
export const load: ServerLoad = async ({ locals: { guard } }) => {
const user = guard.requiresAuth().orRedirects().getUser();
return {
sessions: getUserSessions(user.id),
};
};
export const actions = {
delete: async ({ locals: { guard }, request }) => {
if (guard.requiresAuth().isFailed()) return FORBIDDEN;
const id = (await request.formData()).get('id');
const deleted = deleteSession({ publicId: id?.toString() });
return deleted ? SUCCESS : fail(404, { error: 'Could not find session.' });
},
} satisfies Actions;

View File

@ -0,0 +1,47 @@
<script>
import { browser } from '$app/environment';
import { pageTitle } from '$lib/v2/globalStores.js';
import ListPage from '$lib/v2/snippets/ListPage.svelte';
import { Button, Separator } from 'bits-ui';
import IconPlugConnectedX from '~icons/tabler/plug-connected-x';
let { data, form } = $props();
$pageTitle = 'My active sessions';
$effect(() => {
if (form?.success && browser) {
window.location.reload();
}
});
</script>
<ListPage>
<div class="flex flex-col gap-10 sm:mx-auto sm:w-4/5">
{#each data.sessions as session}
<div
class="flex flex-col gap-5 w-full rounded-xl sm:rounded-2xl border border-neutral-600
px-6 py-5 sm:px-10 sm:py-8 bg-neutral-950 shadow-xl text-neutral-500 text-sm text-justify"
>
<p>User-Agent: <span class="text-neutral-200 font-semibold">{session.userAgent}</span></p>
<p>IP Address: <span class="text-neutral-200 font-semibold">{session.ip}</span></p>
<Separator.Root class="bg-neutral-600 h-px w-full"></Separator.Root>
<div class="flex items-center justify-end gap-5 text-xs sm:text-sm">
<p>Don't recognize this?</p>
<form method="POST" action="?/delete">
<input type="hidden" name="id" value={session.publicId} />
<Button.Root
class="flex items-center justify-center gap-2 bg-neutral-100 text-neutral-950 rounded-full px-4 p-2
border border-white cursor-pointer transition-all ease-in-out duration-300
hover:scale-95 hover:bg-neutral-400"
>
End session
<IconPlugConnectedX />
</Button.Root>
</form>
</div>
</div>
{/each}
<div class="h-10"></div>
</div>
</ListPage>