mirror of
https://github.com/supabase/supabase.git
synced 2026-06-28 19:39:19 -04:00
171ca026b5
Adds integrations settings page to each oauth integration to show associated resources (e.g. API keys, config, oauth apps, etc) <img width="1150" height="892" alt="Screenshot 2026-06-16 at 2 44 31 PM" src="https://github.com/user-attachments/assets/035cc602-886d-43bc-a5a7-e14f76dd37c3" /> <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit ## Summary * **New Features** * Added a Marketplace “Settings” tab with a grouped **Connected resources** view (OAuth apps, API keys, Edge Function secrets, SMTP), including loading/empty/missing-resource states and per-kind removal actions. * Added a resource-group section UI plus integration-aware grouping/copy customization and missing-kind zero-states. * **Bug Fixes** * Improved installed-state detection for Grafana and Doppler by broadening conditions. * Added an orphaned-resources warning when expected OAuth apps are missing. * **Refactor** * Unified connected-resource removal into a single flow with OAuth-specific revoke handling. * **Tests** * Added comprehensive UI and utility coverage for grouping, states, and destructive removal behavior. <!-- end of auto-generated comment: release notes by coderabbit.ai -->
90 lines
3.1 KiB
TypeScript
90 lines
3.1 KiB
TypeScript
import { Settings, Trash2, TriangleAlert } from 'lucide-react'
|
|
import { Badge, Button } from 'ui'
|
|
import { Admonition } from 'ui-patterns'
|
|
|
|
import { type ResourceGroup } from './MarketplaceIntegrationSettingsTab.types'
|
|
import { type ConnectedResource } from '@/components/interfaces/Integrations/Landing/Landing.utils'
|
|
|
|
export const ResourceGroupSection = ({
|
|
group,
|
|
onRemove,
|
|
}: {
|
|
group: ResourceGroup
|
|
onRemove: (resource: ConnectedResource) => void
|
|
}) => {
|
|
return (
|
|
<section className="flex flex-col gap-y-4 border-b py-8 first:pt-0 last:border-b-0">
|
|
<div className="flex flex-col gap-y-2">
|
|
<div className="flex items-center gap-x-2">
|
|
<h3 className="text-base text-foreground">{group.title}</h3>
|
|
{group.missing ? (
|
|
<Badge variant="warning" className="gap-x-1.5">
|
|
<TriangleAlert size={12} strokeWidth={1.5} />
|
|
Not connected
|
|
</Badge>
|
|
) : (
|
|
group.badge && (
|
|
<Badge variant="success" className="gap-x-1.5">
|
|
<span className="h-1.5 w-1.5 rounded-full bg-brand" />
|
|
{group.badge}
|
|
</Badge>
|
|
)
|
|
)}
|
|
</div>
|
|
<p className="text-sm text-foreground-light max-w-2xl">{group.description}</p>
|
|
</div>
|
|
|
|
<Admonition
|
|
type={group.missing ? 'warning' : 'default'}
|
|
className="m-0 max-w-2xl"
|
|
title={group.missing ? 'This resource is missing' : undefined}
|
|
>
|
|
{group.missing ? group.missingNote : group.note}
|
|
</Admonition>
|
|
|
|
{group.missing ? (
|
|
group.manageAction && (
|
|
<div className="max-w-2xl">
|
|
<Button asChild variant="default" icon={<Settings />}>
|
|
<a href={group.manageAction.href}>{group.manageAction.label}</a>
|
|
</Button>
|
|
</div>
|
|
)
|
|
) : (
|
|
<div className="max-w-2xl divide-y rounded-md border bg-surface-100">
|
|
{group.items.map((item) => (
|
|
<div
|
|
key={item.resource.key}
|
|
className="flex items-center justify-between gap-x-4 px-4 py-3"
|
|
>
|
|
<div className="flex min-w-0 flex-col">
|
|
<code
|
|
className="truncate font-mono text-sm text-foreground"
|
|
title={item.identifier}
|
|
>
|
|
{item.identifier}
|
|
</code>
|
|
{item.meta && <span className="text-xs text-foreground-lighter">{item.meta}</span>}
|
|
</div>
|
|
<div className="flex shrink-0 items-center gap-x-2">
|
|
{group.manageAction && (
|
|
<Button asChild variant="default" icon={<Settings />}>
|
|
<a href={group.manageAction.href}>{group.manageAction.label}</a>
|
|
</Button>
|
|
)}
|
|
<Button
|
|
variant="default"
|
|
icon={<Trash2 className="text-foreground-light" />}
|
|
onClick={() => onRemove(item.resource)}
|
|
>
|
|
Remove
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
))}
|
|
</div>
|
|
)}
|
|
</section>
|
|
)
|
|
}
|