* Update Supabase docs URLs to use env variable Co-authored-by: a <a@alaisteryoung.com> * Refactor: Use DOCS_URL constant for documentation links This change centralizes documentation links using a new DOCS_URL constant, improving maintainability and consistency. Co-authored-by: a <a@alaisteryoung.com> * Refactor: Use DOCS_URL constant for all documentation links This change replaces hardcoded documentation URLs with a centralized constant, improving maintainability and consistency. Co-authored-by: a <a@alaisteryoung.com> * replace more instances * ci: Autofix updates from GitHub workflow * remaining instances * fix duplicate useRouter --------- Co-authored-by: Cursor Agent <cursoragent@cursor.com> Co-authored-by: alaister <10985857+alaister@users.noreply.github.com>
132 lines
5.2 KiB
TypeScript
132 lines
5.2 KiB
TypeScript
import { PermissionAction } from '@supabase/shared-types/out/constants'
|
|
import { Check, Webhook } from 'lucide-react'
|
|
|
|
import { ButtonTooltip } from 'components/ui/ButtonTooltip'
|
|
import { DocsButton } from 'components/ui/DocsButton'
|
|
import { useAsyncCheckPermissions } from 'hooks/misc/useCheckPermissions'
|
|
import { DOCS_URL } from 'lib/constants'
|
|
import { Badge, Input, copyToClipboard } from 'ui'
|
|
import { Hook } from './hooks.constants'
|
|
|
|
interface HookCardProps {
|
|
hook: Hook
|
|
onSelect: () => void
|
|
}
|
|
|
|
export const HookCard = ({ hook, onSelect }: HookCardProps) => {
|
|
const { can: canUpdateAuthHook } = useAsyncCheckPermissions(PermissionAction.AUTH_EXECUTE, '*')
|
|
|
|
return (
|
|
<div className="bg-surface-100 border-default overflow-hidden border shadow px-5 py-4 flex flex-row first:rounded-t-md last:rounded-b-md space-x-4">
|
|
<div className="">
|
|
<Webhook size={21} strokeWidth="1" />
|
|
</div>
|
|
<div className="flex flex-col flex-0 overflow-y-auto w-full">
|
|
<span className="text-sm text-foreground">{hook.title}</span>
|
|
<span className="text-sm text-foreground-lighter">{hook.subtitle}</span>
|
|
<div className="text-sm flex flex-row space-x-5 py-4">
|
|
{hook.method.type === 'postgres' ? (
|
|
<div className="flex flex-col w-full space-y-2">
|
|
<div className="flex flex-row items-center">
|
|
<span className="text-foreground-light w-20">type</span>
|
|
<span className="text-foreground">Postgres function</span>
|
|
</div>
|
|
<div className="flex flex-row items-center">
|
|
<span className="text-foreground-light w-20">schema</span>
|
|
<Input
|
|
title={hook.method.schema}
|
|
copy
|
|
readOnly
|
|
disabled
|
|
className="input-mono [&>div>div>div>input]:text-xs [&>div>div>div>input]:opacity-100 flex-1"
|
|
value={hook.method.schema}
|
|
onCopy={() =>
|
|
hook.method.type === 'postgres' && copyToClipboard(hook.method.schema)
|
|
}
|
|
/>
|
|
</div>
|
|
<div className="flex flex-row items-center">
|
|
<span className="text-foreground-light w-20">function</span>
|
|
<Input
|
|
title={hook.method.functionName}
|
|
copy
|
|
readOnly
|
|
disabled
|
|
className="input-mono [&>div>div>div>input]:text-xs [&>div>div>div>input]:opacity-100 flex-1"
|
|
value={hook.method.functionName}
|
|
onCopy={() =>
|
|
hook.method.type === 'postgres' && copyToClipboard(hook.method.functionName)
|
|
}
|
|
/>
|
|
</div>
|
|
</div>
|
|
) : (
|
|
<div className="flex flex-col w-full space-y-2">
|
|
<div className="flex flex-row items-center">
|
|
<span className="text-foreground-light w-20">type</span>
|
|
<span className="text-foreground">HTTPS endpoint</span>
|
|
</div>
|
|
<div className="flex flex-row items-center">
|
|
<span className="text-foreground-light w-20">endpoint</span>
|
|
<Input
|
|
title={hook.method.url}
|
|
copy
|
|
readOnly
|
|
disabled
|
|
className="input-mono [&>div>div>div>input]:text-xs [&>div>div>div>input]:opacity-100 flex-1"
|
|
value={hook.method.url}
|
|
onCopy={() => hook.method.type === 'https' && copyToClipboard(hook.method.url)}
|
|
/>
|
|
</div>
|
|
<div className="flex flex-row items-center">
|
|
<span className="text-foreground-light w-20">secret</span>
|
|
<Input
|
|
copy
|
|
title={hook.method.secret}
|
|
reveal={true}
|
|
readOnly
|
|
disabled
|
|
className="input-mono [&>div>div>div>input]:text-xs [&>div>div>div>input]:opacity-100 flex-1"
|
|
value={hook.method.secret}
|
|
/>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</div>
|
|
<div className="flex flex-row gap-2">
|
|
<ButtonTooltip
|
|
type="default"
|
|
disabled={!canUpdateAuthHook}
|
|
onClick={() => onSelect()}
|
|
tooltip={{
|
|
content: {
|
|
side: 'bottom',
|
|
text: !canUpdateAuthHook
|
|
? 'You need additional permissions to configure auth hooks'
|
|
: undefined,
|
|
},
|
|
}}
|
|
>
|
|
Configure hook
|
|
</ButtonTooltip>
|
|
<DocsButton href={`${DOCS_URL}/guides/auth/auth-hooks/${hook.docSlug}`} />
|
|
</div>
|
|
</div>
|
|
<div className="flex-1">
|
|
{hook.enabled ? (
|
|
<Badge className="space-x-1" variant="brand">
|
|
<div className="h-3.5 w-3.5 bg-brand rounded-full flex justify-center items-center">
|
|
<Check className="h-2 w-2 text-background-overlay " strokeWidth={6} />
|
|
</div>
|
|
<span>Enabled</span>
|
|
</Badge>
|
|
) : (
|
|
<Badge variant="warning">
|
|
<span>Disabled</span>
|
|
</Badge>
|
|
)}
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|