mirror of
https://github.com/amir20/dozzle.git
synced 2025-12-21 21:33:18 +01:00
45 lines
1.6 KiB
Vue
45 lines
1.6 KiB
Vue
<template>
|
|
<div class="relative flex w-full items-start gap-x-2">
|
|
<LogStd :std="logEntry.std" v-if="showStd" />
|
|
<ContainerName class="flex-none" :id="logEntry.containerID" v-if="showContainerName" />
|
|
<LogDate :date="logEntry.date" v-if="showTimestamp" class="select-none" />
|
|
<LogLevel class="flex" :level="logEntry.level" :position="logEntry.position" />
|
|
<div
|
|
class="log-wrapper whitespace-pre-wrap [word-break:break-word] group-[.disable-wrap]:whitespace-nowrap"
|
|
v-html="linkify(colorize(logEntry.message))"
|
|
></div>
|
|
<LogMessageActions
|
|
class="duration-250 absolute -right-1 opacity-0 transition-opacity delay-150 group-hover/entry:opacity-100"
|
|
:message="() => decodeXML(stripAnsi(logEntry.message))"
|
|
:log-entry="logEntry"
|
|
/>
|
|
</div>
|
|
</template>
|
|
<script lang="ts" setup>
|
|
import { SimpleLogEntry } from "@/models/LogEntry";
|
|
import { decodeXML } from "entities";
|
|
import AnsiConvertor from "ansi-to-html";
|
|
import stripAnsi from "strip-ansi";
|
|
|
|
const ansiConvertor = new AnsiConvertor({
|
|
escapeXML: false,
|
|
fg: "var(--base-content-color)",
|
|
bg: "var(--base-color)",
|
|
});
|
|
|
|
const { showContainerName = false } = defineProps<{
|
|
logEntry: SimpleLogEntry;
|
|
showContainerName?: boolean;
|
|
}>();
|
|
|
|
const colorize = (value: string) => ansiConvertor.toHtml(value);
|
|
const urlPattern = /(https?:\/\/[^\s]+)/g;
|
|
const linkify = (text: string) =>
|
|
text.replace(urlPattern, (url) => `<a href="${url}" target="_blank" rel="noopener noreferrer">${url}</a>`);
|
|
</script>
|
|
<style scoped lang="postcss">
|
|
.log-wrapper > :deep(a) {
|
|
@apply text-primary underline-offset-4 hover:underline;
|
|
}
|
|
</style>
|