Files
supabase/apps/studio/components/interfaces/Settings/Logs/LogsFormatters.tsx
Gildas Garcia 96d43099bb chore: refactor Button API so that it can be used a standard button (#46880)
## Problem

Our `<Button>` component breaks the default `button` contract by
redefining the `type` prop to set its variant (`primary`, `default`,
etc) instead of the button type (`submit`, `button`, etc).
This is confusing and forces to write more code when using it with
shadcn components that expect/inject the standard button props.

## Solution

- rename the `type` prop to `variant`
- rename the `htmlType` prop to `type`
- propagate the changes where necessary
- format code

## How to test

As this is just prop renaming, if it builds it's ok

---------

Co-authored-by: Ivan Vasilov <vasilov.ivan@gmail.com>
2026-06-16 23:59:58 +02:00

254 lines
6.5 KiB
TypeScript

/*
* Response Code
*
* for http response codes
*/
import dayjs from 'dayjs'
import { AlertCircle, Info } from 'lucide-react'
import React from 'react'
import { cn } from 'ui'
import { isUnixMicro, unixMicroToIsoTimestamp } from './Logs.utils'
import CopyButton from '@/components/ui/CopyButton'
export const RowLayout: React.FC<React.PropsWithChildren> = ({ children }) => (
<div className="flex h-full w-full items-center gap-4">{children}</div>
)
// renders a timestamp (either unix microsecond or iso timestamp)
export const SelectionDetailedTimestampRow = ({ value }: { value: string | number }) => (
<SelectionDetailedRow
label="Timestamp"
value={isUnixMicro(value) ? unixMicroToIsoTimestamp(value) : String(value)}
/>
)
export const SelectionDetailedRow = ({
label,
value,
valueRender,
}: {
label: string
value: string
valueRender?: React.ReactNode
}) => {
return (
<div className="group flex items-center gap-2 flex-wrap">
<span className="text-foreground-lighter text-sm col-span-3 whitespace-pre-wrap">
{label}
</span>
<span
title={value}
className="truncate font-mono text-foreground text-sm whitespace-pre-wrap break-all"
>
{valueRender ?? value}
</span>
<CopyButton
iconOnly
text={value}
className="group-hover:opacity-100 opacity-0 p-0 h-6 w-6"
variant="text"
title="Copy to clipboard"
/>
</div>
)
}
// used for column renderers
export const TextFormatter: React.FC<{ className?: string; value: string }> = ({
value,
className,
}) => (
<span className={cn('font-mono text-xs truncate select-text cursor-text', className)}>
{value}
</span>
)
export const ResponseCodeFormatter = ({ value }: { value: string }) => {
if (!value) {
return (
<div>
<label className="text-xs text-border-stronger">No data</label>
</div>
)
}
const ResponseCodeItem = ({
children,
className,
}: {
children: React.ReactNode
className?: string
}) => (
<div className="flex h-full items-center">
<div
className={cn(
`relative flex h-6 items-center rounded-md justify-center px-2 py-1 text-center`,
className
)}
>
<label className="block font-mono text-sm">{children}</label>
</div>
</div>
)
const split = value.toString().split('')[0]
switch (split) {
// 2XX || 1XX responses
case '1':
return <ResponseCodeItem>{value}</ResponseCodeItem>
case '2':
return <ResponseCodeItem className="bg-surface-100 text-brand">{value}</ResponseCodeItem>
// 5XX responses
case '5':
return <ResponseCodeItem className="bg-red-300 text-red-1100">{value}</ResponseCodeItem>
// 4XX || 3XX responses
case '4':
case '3':
return <ResponseCodeItem className="bg-warning/10 text-warning">{value}</ResponseCodeItem>
// All other responses
default:
return (
<ResponseCodeItem className="bg-surface-100 text-foreground-lighter">
{value}
</ResponseCodeItem>
)
}
}
/*
* Response Code
*
* for http response codes
*/
export const SeverityFormatter = ({
value,
uppercase = true,
}: {
value: string
uppercase?: boolean
}) => {
if (!value) {
return (
<div>
<label className="text-xs text-border-stronger">No data</label>
</div>
)
}
const uppercasedValue = value.toUpperCase()
const text = uppercase ? uppercasedValue : value
const Layout: React.FC<React.PropsWithChildren<{ className?: string }>> = ({
className,
children,
}) => <div className={`w-24 flex items-center h-full ${className}`}>{children}</div>
switch (uppercasedValue) {
case 'UNCAUGHTEXCEPTION':
case 'PANIC':
case 'FATAL':
case 'ERROR':
return (
<Layout className="gap-1">
<div className=" p-0.5 rounded-sm text-red-900!">
<AlertCircle size={14} strokeWidth={2} />
</div>
<span className="text-red-900! block! titlecase">{text}</span>
</Layout>
)
break
case 'INFO':
case 'DEBUG':
return (
<Layout className="gap-1">
<div className=" p-0.5 rounded-sm text-blue-900!">
<AlertCircle size={14} strokeWidth={2} />
</div>
<span className="text-blue-900! block! titlecase">{text}</span>
</Layout>
)
break
case 'LOG':
return (
<Layout className="gap-1">
<div className=" p-0.5 rounded-sm text-blue-900!">
<Info size={14} strokeWidth={2} />
</div>
<span className="text-blue-900! block! titlecase">{text}</span>
</Layout>
)
break
case 'WARN':
case 'WARNING':
return (
<Layout className="gap-1">
<div className=" p-0.5 rounded-sm text-amber-900!">
<AlertCircle size={14} strokeWidth={2} />
</div>
<span className="text-amber-900! block! titlecase">{text}</span>
</Layout>
)
break
// All other responses
default:
return (
<Layout>
<div className="relative rounded-sm px-2 py-1 text-center h-6 flex justify-center items-center bg-surface-100">
<label className="block font-mono text-sm text-foreground-lighter">{text}</label>
</div>
</Layout>
)
break
}
}
export const timestampLocalFormatter = (value: string | number) => {
const timestamp = isUnixMicro(value) ? unixMicroToIsoTimestamp(value) : value
return dayjs(timestamp).format('DD MMM HH:mm:ss')
}
/*
* JSON Syntax Highlighter
*
* for http response codes
*/
export function jsonSyntaxHighlight(input: Object) {
let json: string = JSON.stringify(input, null, 2)
json = json.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;')
const newJson = json.replace(
/("(\\u[a-zA-Z0-9]{4}|\\[^u]|[^\\"])*"(\s*:)?|\b(true|false|null)\b|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?)/g,
function (match) {
var cls = 'number text-tomato-900'
if (/^"/.test(match)) {
if (/:$/.test(match)) {
cls = 'key text-foreground'
} else {
cls = 'string text-brand-600'
}
} else if (/true|false/.test(match)) {
cls = 'boolean text-blue-900'
} else if (/null/.test(match)) {
cls = 'null text-amber-1100'
}
return '<span class="' + cls + '">' + match + '</span>'
}
)
const jsonWithLineWraps = newJson.split(`\n`).map((x) => {
return `<span class="line text-xs">${x}</span>`
})
return jsonWithLineWraps.join('\n')
}