Files
archtika/web-app/src/routes/(authenticated)/website/[websiteId]/logs/+page.svelte

271 lines
8.6 KiB
Svelte
Raw Normal View History

2024-09-13 17:04:04 +02:00
<script lang="ts">
import WebsiteEditor from "$lib/components/WebsiteEditor.svelte";
import DateTime from "$lib/components/DateTime.svelte";
import Modal from "$lib/components/Modal.svelte";
import type { PageServerData, ActionData } from "./$types";
2024-09-13 17:04:04 +02:00
import { page } from "$app/stores";
import { tables } from "$lib/db-schema";
import { previewContent } from "$lib/runes.svelte";
2024-10-03 18:51:30 +02:00
import DOMPurify from "isomorphic-dompurify";
import { enhanceForm } from "$lib/utils";
import { enhance } from "$app/forms";
import { sending } from "$lib/runes.svelte";
import LoadingSpinner from "$lib/components/LoadingSpinner.svelte";
2024-09-13 17:04:04 +02:00
const { data, form }: { data: PageServerData; form: ActionData } = $props();
2024-09-13 17:04:04 +02:00
let resources = $state({});
if (data.website.content_type === "Blog") {
2024-09-17 22:44:16 +02:00
// eslint-disable-next-line @typescript-eslint/no-unused-vars
2024-09-13 17:04:04 +02:00
const { user, change_log, media, docs_category, ...restTables } = tables;
resources = restTables;
}
if (data.website.content_type === "Docs") {
2024-09-17 22:44:16 +02:00
// eslint-disable-next-line @typescript-eslint/no-unused-vars
2024-09-13 17:04:04 +02:00
const { user, change_log, media, ...restTables } = tables;
resources = restTables;
}
2024-09-15 21:37:32 +02:00
previewContent.value = data.home.main_content;
2024-09-13 17:04:04 +02:00
</script>
{#if sending.value}
<LoadingSpinner />
{/if}
2024-09-13 17:04:04 +02:00
<WebsiteEditor
id={data.website.id}
contentType={data.website.content_type}
title={data.website.title}
>
<section id="logs">
2024-09-13 17:04:04 +02:00
<hgroup>
<h2>
<a href="#logs">Logs</a>
</h2>
<p>
2024-09-15 21:37:32 +02:00
<strong>{data.resultChangeLogCount.toLocaleString("en", { useGrouping: true })}</strong>
<small>result(s)</small>
2024-09-13 17:04:04 +02:00
</p>
</hgroup>
<details>
<summary>Filter</summary>
<form method="GET">
<label>
Username:
<input
list="users-{data.website.id}"
name="logs_filter_user"
value={$page.url.searchParams.get("logs_filter_user")}
/>
<datalist id="users-{data.website.id}">
<option value={data.website.user.username}></option>
{#each data.collaborators as { user: { username } }}
<option value={username}></option>
{/each}
</datalist>
</label>
<label>
Resource:
<select name="logs_filter_resource">
<option value="all">Show all</option>
{#each Object.keys(resources) as resource}
<option
value={resource}
selected={resource === $page.url.searchParams.get("logs_filter_resource")}
>{resource}</option
>
{/each}
</select>
</label>
<label>
Operation:
<select name="logs_filter_operation">
<option value="all">Show all</option>
<option
value="insert"
selected={"insert" === $page.url.searchParams.get("logs_filter_operation")}
>Insert</option
>
<option
value="update"
selected={"update" === $page.url.searchParams.get("logs_filter_operation")}
>Update</option
>
<option
value="delete"
selected={"delete" === $page.url.searchParams.get("logs_filter_operation")}
>Delete</option
>
</select>
</label>
2024-09-15 21:37:32 +02:00
<input type="hidden" name="logs_results_page" value={1} />
2024-09-13 17:04:04 +02:00
<button type="submit">Submit</button>
</form>
</details>
<div class="scroll-container">
<table>
<thead>
<tr>
<th>User</th>
<th>Resource</th>
<th>Operation</th>
<th>Date and time</th>
<th>Changes</th>
</tr>
</thead>
<tbody>
{#each data.changeLog as { id, table_name, operation, tstamp, old_value, new_value, user_id, username }}
2024-09-13 17:04:04 +02:00
<tr>
<td>
<span style:text-decoration={user_id ? "" : "line-through"}>
{username}
</span>
</td>
2024-09-13 17:04:04 +02:00
<td>{table_name}</td>
<td>{operation}</td>
<td>
<DateTime date={tstamp} />
</td>
<td>
<Modal id="log-{id}" text="Show" isWider={true}>
{@const oldValue = JSON.stringify(old_value, null, 2)}
{@const newValue = JSON.stringify(new_value, null, 2)}
<hgroup>
<h3>Log changes</h3>
<p>{table_name} &mdash; {operation} &mdash; User "{username}"</p>
2024-09-13 17:04:04 +02:00
</hgroup>
{#if old_value && new_value}
<h4>Difference</h4>
<form action="?/computeDiff" method="POST" use:enhance={enhanceForm()}>
<input type="hidden" name="id" value={id} />
<button type="submit">Compute diff</button>
</form>
{#if form?.logId === id && form?.currentDiff}
<pre style="white-space: pre-wrap">{@html DOMPurify.sanitize(
form.currentDiff,
{ ALLOWED_TAGS: ["ins", "del"] }
)}</pre>
{/if}
{/if}
{#if new_value && !old_value}
<h4>New value</h4>
<pre style="white-space: pre-wrap">{DOMPurify.sanitize(newValue)}</pre>
{/if}
{#if old_value && !new_value}
<h4>Old value</h4>
<pre style="white-space: pre-wrap">{DOMPurify.sanitize(oldValue)}</pre>
{/if}
2024-09-13 17:04:04 +02:00
</Modal>
</td>
</tr>
{/each}
</tbody>
</table>
2024-09-17 21:27:41 +02:00
</div>
<div class="pagination">
2024-09-15 21:37:32 +02:00
{#snippet commonFilterInputs()}
<input
type="hidden"
name="logs_filter_user"
value={$page.url.searchParams.get("logs_filter_user")}
/>
<input
type="hidden"
name="logs_filter_resource"
value={$page.url.searchParams.get("logs_filter_resource")}
/>
<input
type="hidden"
name="logs_filter_operation"
value={$page.url.searchParams.get("logs_filter_operation")}
/>
{/snippet}
2024-09-17 21:27:41 +02:00
<p>
{$page.url.searchParams.get("logs_results_page") ?? 1} / {Math.max(
Math.ceil(data.resultChangeLogCount / 20),
2024-09-17 21:27:41 +02:00
1
)}
</p>
<form method="GET">
<input type="hidden" name="logs_results_page" value={1} />
{@render commonFilterInputs()}
<button
type="submit"
disabled={($page.url.searchParams.get("logs_results_page") ?? "1") === "1"}>First</button
>
</form>
<form method="GET">
<input
type="hidden"
name="logs_results_page"
value={Math.max(
1,
Number.parseInt($page.url.searchParams.get("logs_results_page") ?? "1") - 1
2024-09-15 21:37:32 +02:00
)}
2024-09-17 21:27:41 +02:00
/>
{@render commonFilterInputs()}
<button
type="submit"
disabled={($page.url.searchParams.get("logs_results_page") ?? "1") === "1"}
>Previous</button
>
</form>
<form method="GET">
<input
type="hidden"
name="logs_results_page"
value={Math.min(
Math.max(Math.ceil(data.resultChangeLogCount / 20), 1),
2024-09-17 21:27:41 +02:00
Number.parseInt($page.url.searchParams.get("logs_results_page") ?? "1") + 1
)}
/>
{@render commonFilterInputs()}
<button
type="submit"
disabled={($page.url.searchParams.get("logs_results_page") ?? "1") ===
Math.max(Math.ceil(data.resultChangeLogCount / 20), 1).toString()}>Next</button
2024-09-17 21:27:41 +02:00
>
</form>
<form method="GET">
<input
type="hidden"
name="logs_results_page"
value={Math.max(Math.ceil(data.resultChangeLogCount / 20), 1)}
2024-09-17 21:27:41 +02:00
/>
{@render commonFilterInputs()}
<button
type="submit"
disabled={($page.url.searchParams.get("logs_results_page") ?? "1") ===
Math.max(Math.ceil(data.resultChangeLogCount / 20), 1).toString()}>Last</button
2024-09-17 21:27:41 +02:00
>
</form>
2024-09-13 17:04:04 +02:00
</div>
</section>
</WebsiteEditor>
2024-09-15 21:37:32 +02:00
<style>
.pagination {
display: flex;
align-items: center;
flex-wrap: wrap;
gap: var(--space-xs);
justify-content: end;
}
.pagination > form:first-of-type {
margin-inline-start: auto;
}
2024-09-17 22:44:16 +02:00
button:disabled {
2024-09-15 21:37:32 +02:00
pointer-events: none;
2024-09-17 22:44:16 +02:00
color: hsl(0 0% 50%);
2024-09-15 21:37:32 +02:00
}
</style>