mirror of
https://github.com/supabase/supabase.git
synced 2026-05-06 08:56:46 -04:00
8a5e609936
This change updates the handler for simple GET installations to correctly appending the organization_slug and project_ref parameters as documented in our Partner Integrations Guide. Fixes INT-111 <!-- This is an auto-generated comment: release notes by coderabbit.ai --> ## Summary by CodeRabbit * **Bug Fixes** * Integration installs now open in a new browser tab instead of replacing the current page. * Organization and project context are appended to integration installation links so setups receive correct context. * Missing or invalid installation links now fall back to the home page to avoid navigation errors. <!-- end of auto-generated comment: release notes by coderabbit.ai --> --------- Co-authored-by: Joshen Lim <joshenlimek@gmail.com>
93 lines
3.0 KiB
TypeScript
93 lines
3.0 KiB
TypeScript
import { useParams } from 'common'
|
|
import { useMemo } from 'react'
|
|
import { toast } from 'sonner'
|
|
import { Button } from 'ui'
|
|
|
|
import type { IntegrationDefinition } from '@/components/interfaces/Integrations/Landing/Integrations.constants'
|
|
import { useAPIKeysQuery } from '@/data/api-keys/api-keys-query'
|
|
import { useInstallOAuthIntegrationMutation } from '@/data/marketplace/install-oauth-integration-mutation'
|
|
import { useSelectedOrganizationQuery } from '@/hooks/misc/useSelectedOrganization'
|
|
|
|
interface InstallOAuthIntegrationButtonProps {
|
|
integration: IntegrationDefinition
|
|
}
|
|
|
|
export function InstallOAuthIntegrationButton({ integration }: InstallOAuthIntegrationButtonProps) {
|
|
const { ref: projectRef } = useParams()
|
|
const { data: selectedOrg } = useSelectedOrganizationQuery()
|
|
|
|
const { data: apiKeys, isLoading: isApiKeysLoading } = useAPIKeysQuery(
|
|
{ projectRef, reveal: false },
|
|
{ enabled: !!projectRef }
|
|
)
|
|
|
|
const { mutate: installOAuthIntegration, isPending: isInstalling } =
|
|
useInstallOAuthIntegrationMutation({
|
|
onSuccess: (data) => {
|
|
if ('redirectUrl' in data) {
|
|
if (!data.redirectUrl) {
|
|
toast.error('Failed to redirect because redirect URL is invalid')
|
|
return
|
|
}
|
|
window.open(data.redirectUrl, '_blank', 'noreferrer')
|
|
} else {
|
|
toast.error('Failed to start integration installation')
|
|
}
|
|
},
|
|
})
|
|
|
|
const isLoading =
|
|
integration.installIdentificationMethod === 'secret_key_prefix' && isApiKeysLoading
|
|
|
|
const isIntegrationInstalled = useMemo(() => {
|
|
if (!integration) return false
|
|
|
|
const prefix = integration.secretKeyPrefix
|
|
|
|
if (integration.installIdentificationMethod !== 'secret_key_prefix' || !prefix) return false
|
|
if (isApiKeysLoading || !apiKeys) return false
|
|
|
|
return apiKeys.some((k) => k.type === 'secret' && k.name.startsWith(prefix))
|
|
}, [apiKeys, integration, isApiKeysLoading])
|
|
|
|
const handleInstallClick = async () => {
|
|
if (!integration || !projectRef) return
|
|
|
|
if (integration.installUrlType === 'post') {
|
|
if (!integration.listingId) return toast.error('Listing ID is required')
|
|
installOAuthIntegration({ projectRef, id: integration.listingId })
|
|
} else {
|
|
let redirectUrl = '/'
|
|
|
|
if (integration.installUrl) {
|
|
const url = new URL(integration.installUrl)
|
|
url.searchParams.append('organization_slug', selectedOrg?.slug ?? '')
|
|
url.searchParams.append('project_id', projectRef)
|
|
redirectUrl = url.href
|
|
}
|
|
|
|
window.open(redirectUrl, '_blank', 'noreferrer')
|
|
}
|
|
}
|
|
|
|
return (
|
|
<>
|
|
{isIntegrationInstalled ? (
|
|
<Button disabled type="outline" className="shrink-0">
|
|
Installed
|
|
</Button>
|
|
) : (
|
|
<Button
|
|
type="primary"
|
|
className="shrink-0"
|
|
loading={isInstalling || isLoading}
|
|
disabled={isLoading}
|
|
onClick={handleInstallClick}
|
|
>
|
|
Install integration
|
|
</Button>
|
|
)}
|
|
</>
|
|
)
|
|
}
|