Files
supabase/apps/studio/components/interfaces/Settings/API/ServiceList.tsx
Drake Costa 803890f4f9 Update /project/_/settings/api page for consistency with other Project Settings pages (#37996)
* refactor api-settings page

* Minor fix

* nit spacing

* Improve loading

* Improve loading

---------

Co-authored-by: Joshen Lim <joshenlimek@gmail.com>
2025-08-18 23:42:41 +08:00

107 lines
4.3 KiB
TypeScript

import { AlertCircle } from 'lucide-react'
import { useParams } from 'common'
import { ScaffoldSection } from 'components/layouts/Scaffold'
import DatabaseSelector from 'components/ui/DatabaseSelector'
import { GenericSkeletonLoader } from 'components/ui/ShimmeringLoader'
import { useCustomDomainsQuery } from 'data/custom-domains/custom-domains-query'
import { useLoadBalancersQuery } from 'data/read-replicas/load-balancers-query'
import { useReadReplicasQuery } from 'data/read-replicas/replicas-query'
import { useSelectedProjectQuery } from 'hooks/misc/useSelectedProject'
import { PROJECT_STATUS } from 'lib/constants'
import { useDatabaseSelectorStateSnapshot } from 'state/database-selector'
import { Alert_Shadcn_, AlertTitle_Shadcn_, Badge, Card, CardContent, CardHeader } from 'ui'
import { Input } from 'ui-patterns/DataInputs/Input'
import { FormLayout } from 'ui-patterns/form/Layout/FormLayout'
import { PostgrestConfig } from './PostgrestConfig'
export const ServiceList = () => {
const { data: project, isLoading } = useSelectedProjectQuery()
const { ref: projectRef } = useParams()
const state = useDatabaseSelectorStateSnapshot()
const { data: customDomainData } = useCustomDomainsQuery({ projectRef })
const {
data: databases,
isError,
isLoading: isLoadingDatabases,
} = useReadReplicasQuery({ projectRef })
const { data: loadBalancers } = useLoadBalancersQuery({ projectRef })
// Get the API service
const isCustomDomainActive = customDomainData?.customDomain?.status === 'active'
const selectedDatabase = databases?.find((db) => db.identifier === state.selectedDatabaseId)
const loadBalancerSelected = state.selectedDatabaseId === 'load-balancer'
const replicaSelected = selectedDatabase?.identifier !== projectRef
const endpoint =
isCustomDomainActive && state.selectedDatabaseId === projectRef
? `https://${customDomainData.customDomain.hostname}`
: loadBalancerSelected
? loadBalancers?.[0].endpoint ?? ''
: selectedDatabase?.restUrl
return (
<ScaffoldSection isFullWidth id="api-settings" className="gap-6">
{!isLoading && project?.status !== PROJECT_STATUS.ACTIVE_HEALTHY ? (
<Alert_Shadcn_ variant="destructive">
<AlertCircle size={16} />
<AlertTitle_Shadcn_>
API settings are unavailable as the project is not active
</AlertTitle_Shadcn_>
</Alert_Shadcn_>
) : (
<>
<Card>
<CardHeader className="flex flex-row items-center justify-between">
Project URL
<DatabaseSelector
additionalOptions={
(loadBalancers ?? []).length > 0
? [{ id: 'load-balancer', name: 'API Load Balancer' }]
: []
}
/>
</CardHeader>
<CardContent>
{isLoading || isLoadingDatabases ? (
<GenericSkeletonLoader />
) : isError ? (
<Alert_Shadcn_ variant="destructive">
<AlertCircle size={16} />
<AlertTitle_Shadcn_>Failed to retrieve project URL</AlertTitle_Shadcn_>
</Alert_Shadcn_>
) : (
<FormLayout
layout="horizontal"
label={
isCustomDomainActive ? (
<div className="flex items-center space-x-2">
<p>URL</p>
<Badge>Custom domain active</Badge>
</div>
) : (
'URL'
)
}
description={
loadBalancerSelected
? 'RESTful endpoint for querying and managing your databases through your load balancer'
: replicaSelected
? 'RESTful endpoint for querying your read replica'
: 'RESTful endpoint for querying and managing your database'
}
>
<Input copy readOnly disabled className="input-mono" value={endpoint} />
</FormLayout>
)}
</CardContent>
</Card>
<PostgrestConfig />
</>
)}
</ScaffoldSection>
)
}