Co-authored-by: Andrew Bastin <andrewbastin.k@gmail.com> Co-authored-by: Joel Jacob Stephen <70131076+JoelJacobStephen@users.noreply.github.com> Co-authored-by: nivedin <nivedinp@gmail.com>
154 lines
5.0 KiB
Vue
154 lines
5.0 KiB
Vue
<template>
|
|
<div
|
|
class="sticky top-0 z-10 flex flex-shrink-0 items-center justify-center overflow-auto overflow-x-auto whitespace-nowrap bg-primary p-4"
|
|
>
|
|
<AppShortcutsPrompt v-if="response == null" class="flex-1" />
|
|
<div v-else class="flex flex-1 flex-col">
|
|
<div
|
|
v-if="response.type === 'loading'"
|
|
class="flex flex-col items-center justify-center"
|
|
>
|
|
<HoppSmartSpinner class="my-4" />
|
|
<span class="text-secondaryLight">{{ t("state.loading") }}</span>
|
|
</div>
|
|
<HoppSmartPlaceholder
|
|
v-if="response.type === 'network_fail'"
|
|
:src="`/images/states/${colorMode.value}/youre_lost.svg`"
|
|
:alt="`${t('error.network_fail')}`"
|
|
:heading="t('error.network_fail')"
|
|
:text="t('helpers.network_fail')"
|
|
large
|
|
>
|
|
<AppInterceptor class="rounded border border-dividerLight p-2" />
|
|
</HoppSmartPlaceholder>
|
|
<HoppSmartPlaceholder
|
|
v-if="response.type === 'script_fail'"
|
|
:src="`/images/states/${colorMode.value}/youre_lost.svg`"
|
|
:alt="`${t('error.script_fail')}`"
|
|
:label="t('error.script_fail')"
|
|
:text="t('helpers.script_fail')"
|
|
large
|
|
>
|
|
<div
|
|
class="mt-2 w-full overflow-auto whitespace-normal rounded bg-primaryLight px-4 py-2 font-mono text-red-400"
|
|
>
|
|
{{ response.error.name }}: {{ response.error.message }}<br />
|
|
{{ response.error.stack }}
|
|
</div>
|
|
</HoppSmartPlaceholder>
|
|
<div
|
|
v-if="response.type === 'success' || response.type === 'fail'"
|
|
class="flex items-center text-tiny font-semibold"
|
|
>
|
|
<div
|
|
:class="statusCategory.className"
|
|
class="inline-flex flex-1 space-x-4"
|
|
>
|
|
<span v-if="response.statusCode">
|
|
<span class="text-secondary"> {{ t("response.status") }}: </span>
|
|
{{ `${response.statusCode}\xA0 • \xA0`
|
|
}}{{ getStatusCodeReasonPhrase(response.statusCode) }}
|
|
</span>
|
|
<span v-if="response.meta && response.meta.responseDuration">
|
|
<span class="text-secondary"> {{ t("response.time") }}: </span>
|
|
{{ `${response.meta.responseDuration} ms` }}
|
|
</span>
|
|
<span
|
|
v-if="response.meta && response.meta.responseSize"
|
|
v-tippy="
|
|
readableResponseSize
|
|
? { theme: 'tooltip' }
|
|
: { onShow: () => false }
|
|
"
|
|
:title="`${response.meta.responseSize} B`"
|
|
>
|
|
<span class="text-secondary"> {{ t("response.size") }}: </span>
|
|
{{
|
|
readableResponseSize
|
|
? readableResponseSize
|
|
: `${response.meta.responseSize} B`
|
|
}}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<AppInspection
|
|
v-if="response?.type !== 'loading'"
|
|
:inspection-results="tabResults"
|
|
:class="[
|
|
response === null || response?.type === 'network_fail'
|
|
? 'absolute right-2 top-2'
|
|
: '-m-2 ml-2',
|
|
]"
|
|
/>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { computed } from "vue"
|
|
import findStatusGroup from "@helpers/findStatusGroup"
|
|
import type { HoppRESTResponse } from "~/helpers/types/HoppRESTResponse"
|
|
import { useI18n } from "@composables/i18n"
|
|
import { useColorMode } from "@composables/theming"
|
|
import { getStatusCodeReasonPhrase } from "~/helpers/utils/statusCodes"
|
|
import { useService } from "dioc/vue"
|
|
import { InspectionService } from "~/services/inspection"
|
|
import { RESTTabService } from "~/services/tab/rest"
|
|
|
|
const t = useI18n()
|
|
const colorMode = useColorMode()
|
|
const tabs = useService(RESTTabService)
|
|
|
|
const props = defineProps<{
|
|
response: HoppRESTResponse | null | undefined
|
|
}>()
|
|
|
|
/**
|
|
* Gives the response size in a human readable format
|
|
* (changes unit from B to MB/KB depending on the size)
|
|
* If no changes (error res state) or value can be made (size < 1KB ?),
|
|
* it returns undefined
|
|
*/
|
|
const readableResponseSize = computed(() => {
|
|
if (
|
|
props.response === null ||
|
|
props.response === undefined ||
|
|
props.response.type === "loading" ||
|
|
props.response.type === "network_fail" ||
|
|
props.response.type === "script_fail" ||
|
|
props.response.type === "fail"
|
|
)
|
|
return undefined
|
|
|
|
const size = props.response.meta.responseSize
|
|
|
|
if (size >= 100000) return (size / 1000000).toFixed(2) + " MB"
|
|
if (size >= 1000) return (size / 1000).toFixed(2) + " KB"
|
|
|
|
return undefined
|
|
})
|
|
|
|
const statusCategory = computed(() => {
|
|
if (
|
|
props.response === null ||
|
|
props.response === undefined ||
|
|
props.response.type === "loading" ||
|
|
props.response.type === "network_fail" ||
|
|
props.response.type === "script_fail" ||
|
|
props.response.type === "fail"
|
|
)
|
|
return {
|
|
name: "error",
|
|
className: "text-red-500",
|
|
}
|
|
return findStatusGroup(props.response.statusCode)
|
|
})
|
|
|
|
const inspectionService = useService(InspectionService)
|
|
|
|
const tabResults = inspectionService.getResultViewFor(
|
|
tabs.currentTabID.value,
|
|
(result) => result.locations.type === "response"
|
|
)
|
|
</script>
|