Files
supabase/apps/studio/components/layouts/ProjectNeedsSecuring/ProjectNeedsSecuringView.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

240 lines
8.3 KiB
TypeScript

import { ArrowRight, Check, ExternalLink, Lightbulb, X } from 'lucide-react'
import Link from 'next/link'
import { useRouter } from 'next/router'
import { useMemo } from 'react'
import {
Button,
Button_Shadcn_,
Card,
Table,
TableBody,
TableCell,
TableHead,
TableHeader,
TableRow,
} from 'ui'
import { PageContainer } from 'ui-patterns/PageContainer'
import {
PageHeader,
PageHeaderAside,
PageHeaderDescription,
PageHeaderIcon,
PageHeaderMeta,
PageHeaderSummary,
PageHeaderTitle,
} from 'ui-patterns/PageHeader'
import {
PageSection,
PageSectionAside,
PageSectionContent,
PageSectionMeta,
PageSectionSummary,
PageSectionTitle,
} from 'ui-patterns/PageSection'
import { GenericSkeletonLoader } from 'ui-patterns/ShimmeringLoader'
import type {
ProjectSecurityActionDetails,
ProjectSecurityActionType,
ProjectSecurityTable,
} from './ProjectNeedsSecuring.types'
import {
buildSecurityPromptMarkdown,
formatRlsDescription,
getTableKey,
getTablePoliciesHref,
} from './ProjectNeedsSecuring.utils'
import { SIDEBAR_KEYS } from '@/components/layouts/ProjectLayout/LayoutSidebar/LayoutSidebarProvider'
import { AiAssistantDropdown } from '@/components/ui/AiAssistantDropdown'
import AlertError from '@/components/ui/AlertError'
import { createNavigationHandler } from '@/lib/navigation'
import { useAiAssistantStateSnapshot } from '@/state/ai-assistant-state'
import { useSidebarManagerSnapshot } from '@/state/sidebar-manager-state'
const StatusCell = ({ enabled, label }: { enabled: boolean; label: string }) => (
<div className="flex items-center gap-2 text-sm">
{enabled ? (
<Check size={14} className="text-brand" aria-hidden="true" />
) : (
<X size={14} className="text-destructive" aria-hidden="true" />
)}
<span>{label}</span>
</div>
)
export const ProjectNeedsSecuringView = ({
projectRef,
issueCount,
tables,
isLoading,
error,
onDismiss,
onTrackAction,
}: {
projectRef: string
issueCount: number
tables: ProjectSecurityTable[]
isLoading: boolean
error?: { message: string } | null
onDismiss: () => void
onTrackAction: (type: ProjectSecurityActionType, details?: ProjectSecurityActionDetails) => void
}) => {
const router = useRouter()
const aiSnap = useAiAssistantStateSnapshot()
const { openSidebar } = useSidebarManagerSnapshot()
const promptMarkdown = useMemo(
() => buildSecurityPromptMarkdown(issueCount, tables),
[issueCount, tables]
)
const handleOpenAssistant = () => {
onTrackAction('ask_assistant')
openSidebar(SIDEBAR_KEYS.AI_ASSISTANT)
aiSnap.newChat({
name: 'Review project security',
initialInput: promptMarkdown,
})
}
return (
<div className="flex flex-1 flex-col overflow-y-auto">
<PageHeader size="default">
<PageHeaderMeta>
<PageHeaderIcon>
<div className="shrink-0 w-14 h-14 relative bg-destructive-200 border border-destructive-400 rounded-md flex items-center justify-center">
<Lightbulb size={20} strokeWidth={1.5} className="text-destructive" />
</div>
</PageHeaderIcon>
<PageHeaderSummary>
<PageHeaderTitle>Your project needs securing</PageHeaderTitle>
<PageHeaderDescription>{formatRlsDescription(issueCount)}</PageHeaderDescription>
</PageHeaderSummary>
<PageHeaderAside>
<Button asChild variant="text" iconRight={<ArrowRight />}>
<Link
href={`/project/${projectRef}`}
onClick={() => {
onTrackAction('skip_to_home')
onDismiss()
}}
>
Skip to home
</Link>
</Button>
</PageHeaderAside>
</PageHeaderMeta>
</PageHeader>
<PageContainer size="default" className="pb-12">
<PageSection>
<PageSectionMeta>
<PageSectionSummary>
<PageSectionTitle>Review and fix</PageSectionTitle>
</PageSectionSummary>
<PageSectionAside>
<AiAssistantDropdown
label="Ask Assistant"
size="tiny"
buildPrompt={() => promptMarkdown}
onOpenAssistant={handleOpenAssistant}
onCopyPrompt={() => onTrackAction('copy_prompt')}
copyLabel="Copy Markdown"
disabled={isLoading}
/>
</PageSectionAside>
</PageSectionMeta>
<PageSectionContent>
{isLoading ? (
<GenericSkeletonLoader />
) : error ? (
<AlertError
projectRef={projectRef}
error={error}
subject="Failed to retrieve project tables"
/>
) : (
<Card>
<Table>
<TableHeader>
<TableRow>
<TableHead>Name</TableHead>
<TableHead>Schema</TableHead>
<TableHead>
<div className="flex items-center gap-1.5">
<span>Accessible via Data API</span>
<Button_Shadcn_ asChild variant="ghost" size="icon" className="h-6 w-6">
<Link
href={`/project/${projectRef}/integrations/data_api/settings`}
target="_blank"
rel="noreferrer"
aria-label="Open Data API settings"
>
<ExternalLink size={14} aria-hidden="true" />
</Link>
</Button_Shadcn_>
</div>
</TableHead>
<TableHead>RLS</TableHead>
</TableRow>
</TableHeader>
<TableBody>
{tables.map((table) => {
const policiesHref = getTablePoliciesHref(
projectRef,
table.schema,
table.name
)
const handleNavigation = createNavigationHandler(policiesHref, router)
const trackViewPolicies = () =>
onTrackAction('view_policies', {
schema: table.schema,
tableName: table.name,
})
return (
<TableRow
key={getTableKey(table)}
className="relative cursor-pointer inset-focus"
onClick={(event) => {
trackViewPolicies()
handleNavigation(event)
}}
onAuxClick={(event) => {
if (event.button === 1) trackViewPolicies()
handleNavigation(event)
}}
onKeyDown={(event) => {
if (event.key === 'Enter' || event.key === ' ') trackViewPolicies()
handleNavigation(event)
}}
tabIndex={0}
>
<TableCell className="font-medium">{table.name}</TableCell>
<TableCell>{table.schema}</TableCell>
<TableCell>
<StatusCell
enabled={table.dataApiAccessible}
label={table.dataApiAccessible ? 'Accessible' : 'Not accessible'}
/>
</TableCell>
<TableCell>
<StatusCell
enabled={table.rlsEnabled}
label={table.rlsEnabled ? 'Enabled' : 'Disabled'}
/>
</TableCell>
</TableRow>
)
})}
</TableBody>
</Table>
</Card>
)}
</PageSectionContent>
</PageSection>
</PageContainer>
</div>
)
}