mirror of
https://github.com/CorentinTh/it-tools.git
synced 2025-04-22 15:56:15 -04:00
164 lines
4.7 KiB
Vue
164 lines
4.7 KiB
Vue
<script setup lang="ts">
|
|
import { Plus, Trash } from '@vicons/tabler';
|
|
import { useStorage } from '@vueuse/core';
|
|
import _ from 'lodash';
|
|
|
|
import { arrayToMarkdownTable, computeAverage, computeVariance } from './benchmark-builder.models';
|
|
import DynamicValues from './dynamic-values.vue';
|
|
import { useCopy } from '@/composable/copy';
|
|
|
|
const suites = useStorage('benchmark-builder:suites', [
|
|
{ title: 'Suite 1', data: [5, 10] },
|
|
{ title: 'Suite 2', data: [8, 12] },
|
|
]);
|
|
|
|
const unit = useStorage('benchmark-builder:unit', '');
|
|
|
|
const round = (v: number) => Math.round(v * 1000) / 1000;
|
|
|
|
const results = computed(() => {
|
|
return suites.value
|
|
.map(({ data: dirtyData, title }) => {
|
|
const data = dirtyData.filter(_.isNumber);
|
|
|
|
return {
|
|
title,
|
|
size: data.length,
|
|
mean: computeAverage({ data }),
|
|
variance: computeVariance({ data }),
|
|
};
|
|
})
|
|
.sort((a, b) => a.mean - b.mean)
|
|
.map(({ mean, variance, size, title }, index, suites) => {
|
|
const cleanUnit = unit.value.trim();
|
|
const bestMean: number = suites[0].mean;
|
|
const deltaWithBestMean = mean - bestMean;
|
|
const ratioWithBestMean = bestMean === 0 ? '∞' : round(mean / bestMean);
|
|
|
|
const comparisonValues: string
|
|
= (index !== 0 && bestMean !== mean) ? ` (+${round(deltaWithBestMean)}${cleanUnit} ; x${ratioWithBestMean})` : '';
|
|
|
|
return {
|
|
position: index + 1,
|
|
title,
|
|
mean: `${round(mean)}${cleanUnit}${comparisonValues}`,
|
|
variance: `${round(variance)}${cleanUnit}${cleanUnit ? '²' : ''}`,
|
|
size,
|
|
};
|
|
});
|
|
});
|
|
|
|
const { copy } = useCopy({ createToast: false });
|
|
|
|
const header = {
|
|
title: 'Suite',
|
|
size: 'Samples',
|
|
mean: 'Mean',
|
|
variance: 'Variance',
|
|
position: 'Position',
|
|
};
|
|
|
|
function copyAsMarkdown() {
|
|
copy(arrayToMarkdownTable({ data: results.value, headerMap: header }));
|
|
}
|
|
|
|
function copyAsBulletList() {
|
|
const bulletList = results.value
|
|
.flatMap(({ title, ...sections }) => {
|
|
return [
|
|
` - ${title}`,
|
|
...Object.entries(sections).map(
|
|
([key, value]) => ` - ${header[key as keyof typeof header] ?? key}: ${value}`,
|
|
),
|
|
];
|
|
})
|
|
.join('\n');
|
|
|
|
copy(bulletList);
|
|
}
|
|
</script>
|
|
|
|
<template>
|
|
<n-scrollbar style="flex: 1" x-scrollable>
|
|
<div mb-5 flex flex-1 flex-nowrap justify-center gap-12px>
|
|
<div v-for="(suite, index) of suites" :key="index">
|
|
<c-card style="width: 294px">
|
|
<c-input-text
|
|
v-model:value="suite.title"
|
|
label-position="left"
|
|
label="Suite name"
|
|
placeholder="Suite name..."
|
|
clearable
|
|
/>
|
|
|
|
<n-divider />
|
|
<n-form-item label="Suite values" :show-feedback="false">
|
|
<DynamicValues v-model:values="suite.data" />
|
|
</n-form-item>
|
|
</c-card>
|
|
|
|
<div flex justify-center>
|
|
<c-button v-if="suites.length > 1" variant="text" @click="suites.splice(index, 1)">
|
|
<n-icon :component="Trash" depth="3" mr-2 size="18" />
|
|
Delete suite
|
|
</c-button>
|
|
<c-button
|
|
variant="text"
|
|
@click="suites.splice(index + 1, 0, { data: [0], title: `Suite ${suites.length + 1}` })"
|
|
>
|
|
<n-icon :component="Plus" depth="3" mr-2 size="18" />
|
|
Add suite
|
|
</c-button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</n-scrollbar>
|
|
|
|
<div style="flex: 0 0 100%">
|
|
<div style="max-width: 600px; margin: 0 auto">
|
|
<div mx-auto max-w-sm flex justify-center gap-3>
|
|
<c-input-text v-model:value="unit" placeholder="Unit (eg: ms)" label="Unit" label-position="left" mb-4 />
|
|
|
|
<c-button
|
|
@click="
|
|
suites = [
|
|
{ title: 'Suite 1', data: [] },
|
|
{ title: 'Suite 2', data: [] },
|
|
]
|
|
"
|
|
>
|
|
Reset suites
|
|
</c-button>
|
|
</div>
|
|
|
|
<n-table>
|
|
<thead>
|
|
<tr>
|
|
<th>{{ header.position }}</th>
|
|
<th>{{ header.title }}</th>
|
|
<th>{{ header.size }}</th>
|
|
<th>{{ header.mean }}</th>
|
|
<th>{{ header.variance }}</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<tr v-for="{ title, size, mean, variance, position } of results" :key="title">
|
|
<td>{{ position }}</td>
|
|
<td>{{ title }}</td>
|
|
<td>{{ size }}</td>
|
|
<td>{{ mean }}</td>
|
|
<td>{{ variance }}</td>
|
|
</tr>
|
|
</tbody>
|
|
</n-table>
|
|
<div mt-5 flex justify-center gap-3>
|
|
<c-button @click="copyAsMarkdown()">
|
|
Copy as markdown table
|
|
</c-button>
|
|
<c-button @click="copyAsBulletList()">
|
|
Copy as bullet list
|
|
</c-button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|