Files
supabase/apps/studio/components/interfaces/Integrations/Integration/ConnectedResourceGroupSection.tsx
Matt Linkous 171ca026b5 feat(studio): Add integration settings page with connected resources (#46961)
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 -->
2026-06-22 20:51:14 +00:00

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>
)
}