OAuthApps() — supabase Function Reference
Architecture documentation for the OAuthApps() function in OAuthApps.tsx from the supabase codebase.
Entity Profile
Relationship Graph
Source Code
apps/studio/components/interfaces/Organization/OAuthApps/OAuthApps.tsx lines 29–268
export const OAuthApps = () => {
const { slug } = useParams()
const [showPublishModal, setShowPublishModal] = useState(false)
const [createdApp, setCreatedApp] = useState<OAuthAppCreateResponse>()
const [selectedAppToUpdate, setSelectedAppToUpdate] = useState<OAuthApp>()
const [selectedAppToDelete, setSelectedAppToDelete] = useState<OAuthApp>()
const [selectedAppToRevoke, setSelectedAppToRevoke] = useState<AuthorizedApp>()
const { can: canReadOAuthApps, isLoading: isLoadingPermissions } = useAsyncCheckPermissions(
PermissionAction.READ,
'approved_oauth_apps'
)
const { can: canCreateOAuthApps } = useAsyncCheckPermissions(
PermissionAction.CREATE,
'approved_oauth_apps'
)
const {
data: publishedApps,
error: publishedAppsError,
isPending: isLoadingPublishedApps,
isSuccess: isSuccessPublishedApps,
isError: isErrorPublishedApps,
} = useOAuthAppsQuery({ slug }, { enabled: canReadOAuthApps })
const sortedPublishedApps = publishedApps?.sort((a, b) => {
return Number(new Date(a.created_at ?? '')) - Number(new Date(b.created_at ?? ''))
})
const {
data: authorizedApps,
isPending: isLoadingAuthorizedApps,
isSuccess: isSuccessAuthorizedApps,
isError: isErrorAuthorizedApps,
} = useAuthorizedAppsQuery({ slug })
const sortedAuthorizedApps = authorizedApps?.sort((a, b) => {
return Number(new Date(a.authorized_at)) - Number(new Date(b.authorized_at))
})
return (
<>
<ScaffoldContainer>
<ScaffoldSection isFullWidth className="flex flex-col gap-y-8">
<div className="space-y-4">
<div className="flex flex-col md:flex-row items-start md:items-center justify-between gap-2">
<div>
<p>Published Apps</p>
<p className="text-foreground-light text-sm">
Build integrations that extend Supabase's functionality
</p>
</div>
<ButtonTooltip
disabled={!canCreateOAuthApps}
type="primary"
onClick={() => setShowPublishModal(true)}
tooltip={{
content: {
side: 'bottom',
text: !canCreateOAuthApps
? 'You need additional permissions to create apps'
: undefined,
},
}}
>
Add application
</ButtonTooltip>
</div>
{isLoadingPublishedApps || isLoadingPermissions ? (
<div className="space-y-2">
<ShimmeringLoader />
<ShimmeringLoader className="w-3/4" />
<ShimmeringLoader className="w-1/2" />
</div>
) : !canReadOAuthApps ? (
<NoPermission resourceText="view OAuth apps" />
) : null}
{isErrorPublishedApps && (
<AlertError
error={publishedAppsError}
subject="Failed to retrieve published OAuth apps"
/>
)}
{createdApp !== undefined && (
<div
className={cn(
'flex items-center justify-between p-4 px-6 border first:rounded-t last:rounded-b',
'bg-background-alternative',
'rounded'
)}
>
<div className="absolute top-4 right-4">
<Button
type="text"
icon={<X size={18} />}
className="px-1"
onClick={() => setCreatedApp(undefined)}
/>
</div>
<div className="w-full space-y-4">
<div className="flex flex-col gap-0">
<div className="flex items-center gap-2">
<Check size={14} className="text-brand" strokeWidth={3} />
<p className="text-sm">You've created your new OAuth application.</p>
</div>
<p className="text-sm text-foreground-light">
Ensure that you store the client secret securely - you will not be able to see
it again.
</p>
</div>
<div className="flex flex-col gap-1">
<div className="flex items-center gap-2">
<p className="text-sm text-foreground-light">Client ID</p>
<p className="font-mono text-sm">{createdApp.client_id}</p>
<CopyButton text={createdApp.client_id} type="default" iconOnly />
</div>
<div className="flex items-center gap-2">
<p className="text-sm text-foreground-light">Client Secret</p>
<p className="font-mono text-sm">{createdApp.client_secret}</p>
<CopyButton text={createdApp.client_secret} type="default" iconOnly />
</div>
</div>
</div>
</div>
)}
{isSuccessPublishedApps && (
<>
{(publishedApps?.length ?? 0) === 0 ? (
<div className="bg-surface-100 border rounded p-4 flex items-center justify-between mt-4">
<p className="prose text-sm">You do not have any published applications yet</p>
</div>
) : (
<Table
head={[
<Table.th key="icon" className="w-[30px]"></Table.th>,
<Table.th key="name">Name</Table.th>,
<Table.th key="client-id">Client ID</Table.th>,
<Table.th key="created-at">Created at</Table.th>,
<Table.th key="delete-action"></Table.th>,
]}
body={
sortedPublishedApps?.map((app) => (
<OAuthAppRow
key={app.id}
app={app}
onSelectEdit={() => {
setShowPublishModal(true)
setSelectedAppToUpdate(app)
}}
onSelectDelete={() => setSelectedAppToDelete(app)}
/>
)) ?? []
}
/>
)}
</>
)}
</div>
<div>
<p>Authorized Apps</p>
<p className="text-foreground-light text-sm">
Applications that have access to your organization's settings and projects
</p>
<div className="mt-4">
{isLoadingAuthorizedApps || isLoadingPermissions ? (
<div className="space-y-2">
<ShimmeringLoader />
<ShimmeringLoader className="w-3/4" />
<ShimmeringLoader className="w-1/2" />
</div>
) : !canReadOAuthApps ? (
<NoPermission resourceText="view authorized apps" />
) : null}
{isErrorAuthorizedApps && <AlertError subject="Failed to retrieve authorized apps" />}
{isSuccessAuthorizedApps && (
<>
{(authorizedApps.length ?? 0) === 0 ? (
<div className="bg-surface-100 border rounded p-4 flex items-center justify-between">
<p className="prose text-sm">
You do not have any authorized applications yet
</p>
</div>
) : (
<Table
className="mt-4"
head={[
<Table.th key="icon" className="w-[30px]"></Table.th>,
<Table.th key="name">Name</Table.th>,
<Table.th key="created-by">Created by</Table.th>,
<Table.th key="app-id">App ID</Table.th>,
<Table.th key="authorized-at">Authorized at</Table.th>,
<Table.th key="delete-action"></Table.th>,
]}
body={
sortedAuthorizedApps?.map((app) => (
<AuthorizedAppRow
key={app.id}
app={app}
onSelectRevoke={() => setSelectedAppToRevoke(app)}
/>
)) ?? []
}
/>
)}
</>
)}
</div>
</div>
</ScaffoldSection>
</ScaffoldContainer>
<PublishAppSidePanel
visible={showPublishModal}
selectedApp={selectedAppToUpdate}
onClose={() => {
setSelectedAppToUpdate(undefined)
setShowPublishModal(false)
}}
onCreateSuccess={setCreatedApp}
/>
<DeleteAppModal
selectedApp={selectedAppToDelete}
onClose={() => setSelectedAppToDelete(undefined)}
/>
<RevokeAppModal
selectedApp={selectedAppToRevoke}
onClose={() => setSelectedAppToRevoke(undefined)}
/>
</>
)
}
Domain
Subdomains
Source
Analyze Your Own Codebase
Get architecture documentation, dependency graphs, and domain analysis for your codebase in minutes.
Try Supermodel Free