vitest/packages/ui/client/components/views/ViewTestReport.vue

252 lines
7.7 KiB
Vue

<script setup lang="ts">
import type { TestArtifactLocation } from '@vitest/runner'
import type { RunnerTestCase } from 'vitest'
import { relative } from 'pathe'
import { computed } from 'vue'
import { getAttachmentUrl, sanitizeFilePath } from '~/composables/attachments'
import { browserState, config } from '~/composables/client'
import { showAttachmentSource } from '~/composables/codemirror'
import { isDark } from '~/composables/dark'
import { mapLeveledTaskStacks } from '~/composables/error'
import { openScreenshot, useScreenshot } from '~/composables/screenshot'
import AnnotationAttachmentImage from '../AnnotationAttachmentImage.vue'
import VisualRegression from '../artifacts/visual-regression/VisualRegression.vue'
import IconButton from '../IconButton.vue'
import Modal from '../Modal.vue'
import ScreenshotError from './ScreenshotError.vue'
import ViewReportError from './ViewReportError.vue'
const props = defineProps<{
test: RunnerTestCase
}>()
const failed = computed(() => {
if (!props.test.result || !props.test.result.errors?.length) {
return null
}
return mapLeveledTaskStacks(isDark.value, [props.test])[0] as RunnerTestCase | null
})
function openLocation(location?: TestArtifactLocation) {
return showAttachmentSource(props.test, location)
}
const {
currentTask,
showScreenshot,
showScreenshotModal,
currentScreenshotUrl,
} = useScreenshot()
function getLocationString(location: TestArtifactLocation) {
const path = relative(config.value.root, location.file)
return `${path}:${location.line}:${location.column}`
}
const kWellKnownMeta = new Set([
'benchmark',
'typecheck',
'failScreenshotPath',
])
const meta = computed(() => {
return Object.entries(props.test.meta).filter(([name]) => {
return !kWellKnownMeta.has(name)
})
})
</script>
<template>
<div h-full class="scrolls">
<div v-if="failed">
<div
bg="red-500/10"
text="red-500 sm"
p="x3 y2"
m-2
rounded
>
<div flex="~ gap-2 items-center">
<template v-if="browserState && test.meta?.failScreenshotPath">
<IconButton
v-tooltip.bottom="'View screenshot error'"
class="!op-100"
icon="i-carbon:image"
title="View screenshot error"
@click="showScreenshotModal(test)"
/>
<IconButton
v-tooltip.bottom="'Open screenshot error in editor'"
class="!op-100"
icon="i-carbon:image-reference"
title="Open screenshot error in editor"
@click="openScreenshot(test)"
/>
</template>
</div>
<div
v-if="test.result?.htmlError"
class="scrolls scrolls-rounded task-error"
data-testid="task-error"
>
<pre v-html="test.result.htmlError" />
</div>
<template v-else-if="test.result?.errors">
<ViewReportError
v-for="(error, idx) of test.result.errors"
:key="idx"
:file-id="test.file.id"
:error="error"
:filename="test.file.name"
:root="config.root"
/>
</template>
</div>
</div>
<template v-else>
<div bg="green-500/10" text="green-500 sm" p="x4 y2" m-2 rounded>
All tests passed in this file
</div>
</template>
<template v-if="test.annotations.length">
<h1 m-2>
Test Annotations
</h1>
<div
v-for="annotation of test.annotations"
:key="annotation.type + annotation.message"
bg="yellow-500/10"
text="yellow-500 sm"
p="x3 y2"
m-2
rounded
role="note"
>
<div flex="~ gap-2 items-center justify-between" overflow-hidden>
<div class="flex gap-2" overflow-hidden>
<span class="font-bold" ws-nowrap truncate>{{ annotation.type }}</span>
<a
v-if="annotation.attachment && !annotation.attachment.contentType?.startsWith('image/')"
class="flex gap-1 items-center text-yellow-500/80 cursor-pointer"
:href="getAttachmentUrl(annotation.attachment)"
:download="sanitizeFilePath(annotation.message, annotation.attachment.contentType)"
>
<span class="i-carbon:download block" />
Download
</a>
</div>
<div>
<span
v-if="annotation.location && annotation.location.file === test.file.filepath"
v-tooltip.bottom="'Open in Editor'"
title="Open in Editor"
class="flex gap-1 text-yellow-500/80 cursor-pointer"
ws-nowrap
@click="openLocation(annotation.location)"
>
{{ getLocationString(annotation.location) }}
</span>
<span
v-else-if="annotation.location && annotation.location.file !== test.file.filepath"
class="flex gap-1 text-yellow-500/80"
ws-nowrap
>
{{ getLocationString(annotation.location) }}
</span>
</div>
</div>
<div
class="scrolls scrolls-rounded task-error"
data-testid="task-error"
>
{{ annotation.message }}
</div>
<AnnotationAttachmentImage :annotation="annotation" />
</div>
</template>
<template v-if="test.artifacts.length">
<h1 m-2>
Test Artifacts
</h1>
<div
v-for="artifact, index of test.artifacts"
:key="artifact.type + index"
bg="yellow-500/10"
text="yellow-500 sm"
p="x3 y2"
m-2
rounded
role="note"
>
<div flex="~ gap-2 items-center justify-between" overflow-hidden>
<div>
<span
v-if="artifact.location && artifact.location.file === test.file.filepath"
v-tooltip.bottom="'Open in Editor'"
title="Open in Editor"
class="flex gap-1 text-yellow-500/80 cursor-pointer"
ws-nowrap
@click="openLocation(artifact.location)"
>
{{ getLocationString(artifact.location) }}
</span>
<span
v-else-if="artifact.location && artifact.location.file !== test.file.filepath"
class="flex gap-1 text-yellow-500/80"
ws-nowrap
>
{{ getLocationString(artifact.location) }}
</span>
</div>
</div>
<VisualRegression v-if="artifact.type === 'internal:toMatchScreenshot' && artifact.kind === 'visual-regression'" :regression="artifact" />
</div>
</template>
<template v-if="meta.length">
<h1 m-2>
Test Meta
</h1>
<div
bg="gray/10"
text="black-100 sm"
p="x3 y2"
m-2
rounded
class="grid grid-cols-1 md:grid-cols-[200px_1fr] gap-2"
overflow-hidden
>
<template v-for="([name, content]) of meta" :key="name">
<div font-bold ws-nowrap truncate py-2>
{{ name }}
</div>
<pre overflow-auto bg="gray/30" rounded p-2>{{ content }}</pre>
</template>
</div>
</template>
<template v-if="browserState">
<Modal v-model="showScreenshot" direction="right">
<template v-if="currentTask">
<Suspense>
<ScreenshotError
:file="currentTask.file.filepath"
:name="currentTask.name"
:url="currentScreenshotUrl"
@close="showScreenshot = false"
/>
</Suspense>
</template>
</Modal>
</template>
</div>
</template>
<style scoped>
.task-error {
--cm-ttc-c-thumb: #ccc;
}
html.dark .task-error {
--cm-ttc-c-thumb: #444;
}
</style>