mirror of
https://github.com/supabase/supabase.git
synced 2026-06-22 03:02:48 +08:00
* chore: update themes and add dark-new theme set * chore: update * add deep dark css theme * fix issue with wrong class * chore: add `bg-studio` as a custom remapped color for studio background * updated surface classes to use a surface-75. * update backgrounds and borders * Update CardButton.tsx * Update NavigationIconButton.tsx * chore: add theme selection * Update package.json * Update index.ts * Update package.json * update alias * Update ThemeSettings.tsx * chore: split up theme support * remove old code * Update index.ts * add back in * Update ThemeSettingsOld.tsx * fix issue with system theme leaking through * reset themes * update imports * packagelock updated * Update ThemeSettingsWithPreferredTheme.tsx * Update system.svg * Update NavigationIconButton.tsx * Update NavigationBar.tsx * update * Update light.css * updated some colors * light mode updated * hacked * prettier --------- Co-authored-by: Francesco Sansalvadore <f.sansalvadore@gmail.com> Co-authored-by: Kevin Grüneberg <k.grueneberg1994@gmail.com>
198 lines
7.3 KiB
TypeScript
198 lines
7.3 KiB
TypeScript
import { PermissionAction } from '@supabase/shared-types/out/constants'
|
|
import { JwtSecretUpdateStatus } from '@supabase/shared-types/out/events'
|
|
import Link from 'next/link'
|
|
import { useState } from 'react'
|
|
import { IconAlertCircle, IconLoader, Input } from 'ui'
|
|
|
|
import { useParams } from 'common/hooks'
|
|
import SimpleCodeBlock from 'components/to-be-cleaned/SimpleCodeBlock'
|
|
import Panel from 'components/ui/Panel'
|
|
import { useJwtSecretUpdatingStatusQuery } from 'data/config/jwt-secret-updating-status-query'
|
|
import { useProjectApiQuery } from 'data/config/project-api-query'
|
|
import { useCheckPermissions } from 'hooks'
|
|
|
|
const generateInitSnippet = (endpoint: string) => ({
|
|
js: `
|
|
import { createClient } from '@supabase/supabase-js'
|
|
|
|
const supabaseUrl = '${endpoint}'
|
|
const supabaseKey = process.env.SUPABASE_KEY
|
|
const supabase = createClient(supabaseUrl, supabaseKey)`,
|
|
dart: `
|
|
const supabaseUrl = '${endpoint}';
|
|
const supabaseKey = String.fromEnvironment('SUPABASE_KEY');
|
|
|
|
Future<void> main() async {
|
|
await Supabase.initialize(url: supabaseUrl, anonKey: supabaseKey);
|
|
runApp(MyApp());
|
|
}`,
|
|
})
|
|
|
|
const APIKeys = () => {
|
|
const { ref: projectRef } = useParams()
|
|
|
|
const availableLanguages = [
|
|
{ name: 'Javascript', key: 'js' },
|
|
{ name: 'Dart', key: 'dart' },
|
|
]
|
|
const [selectedLanguage, setSelectedLanguage] = useState(availableLanguages[0])
|
|
|
|
const {
|
|
data: settings,
|
|
isError: isProjectSettingsError,
|
|
isLoading: isProjectSettingsLoading,
|
|
} = useProjectApiQuery({
|
|
projectRef,
|
|
})
|
|
|
|
const {
|
|
data,
|
|
isError: isJwtSecretUpdateStatusError,
|
|
isLoading: isJwtSecretUpdateStatusLoading,
|
|
} = useJwtSecretUpdatingStatusQuery({ projectRef })
|
|
const jwtSecretUpdateStatus = data?.jwtSecretUpdateStatus
|
|
|
|
const canReadAPIKeys = useCheckPermissions(PermissionAction.READ, 'service_api_keys')
|
|
|
|
// Get the API service
|
|
const apiService = settings?.autoApiService
|
|
const apiKeys = apiService?.service_api_keys ?? []
|
|
|
|
// API keys should not be empty. However it can be populated with a delay on project creation
|
|
const isApiKeysEmpty = apiKeys.length === 0
|
|
const isNotUpdatingJwtSecret =
|
|
jwtSecretUpdateStatus === undefined || jwtSecretUpdateStatus === JwtSecretUpdateStatus.Updated
|
|
|
|
const apiUrl = `${apiService?.protocol ?? 'https'}://${apiService?.endpoint ?? '-'}`
|
|
const anonKey = apiKeys.find((key) => key.tags === 'anon')
|
|
|
|
const clientInitSnippet: any = generateInitSnippet(apiUrl)
|
|
const selectedLanguageSnippet = clientInitSnippet[selectedLanguage.key] ?? 'No snippet available'
|
|
|
|
return (
|
|
<Panel
|
|
title={
|
|
<div className="space-y-3">
|
|
<h5 className="text-base">Project API</h5>
|
|
<p className="text-sm text-foreground-light">
|
|
Your API is secured behind an API gateway which requires an API Key for every request.
|
|
<br />
|
|
You can use the parameters below to use Supabase client libraries.
|
|
</p>
|
|
</div>
|
|
}
|
|
>
|
|
{isProjectSettingsError || isJwtSecretUpdateStatusError ? (
|
|
<div className="flex items-center justify-center py-8 space-x-2">
|
|
<IconAlertCircle size={16} strokeWidth={1.5} />
|
|
<p className="text-sm text-foreground-light">
|
|
{isProjectSettingsError ? 'Failed to retrieve API keys' : 'Failed to update JWT secret'}
|
|
</p>
|
|
</div>
|
|
) : isApiKeysEmpty || isProjectSettingsLoading || isJwtSecretUpdateStatusLoading ? (
|
|
<div className="flex items-center justify-center py-8 space-x-2">
|
|
<IconLoader className="animate-spin" size={16} strokeWidth={1.5} />
|
|
<p className="text-sm text-foreground-light">
|
|
{isProjectSettingsLoading || isApiKeysEmpty
|
|
? 'Retrieving API keys'
|
|
: 'JWT secret is being updated'}
|
|
</p>
|
|
</div>
|
|
) : (
|
|
<>
|
|
<Panel.Content>
|
|
<Input
|
|
label="Project URL"
|
|
readOnly
|
|
copy
|
|
disabled
|
|
className="input-mono"
|
|
value={apiUrl}
|
|
descriptionText="A RESTful endpoint for querying and managing your database."
|
|
layout="horizontal"
|
|
/>
|
|
</Panel.Content>
|
|
<Panel.Content
|
|
className={
|
|
'border-t border-panel-border-interior-light dark:border-panel-border-interior-dark'
|
|
}
|
|
>
|
|
<Input
|
|
readOnly
|
|
disabled
|
|
layout="horizontal"
|
|
className="input-mono"
|
|
// @ts-ignore
|
|
label={
|
|
<div className="space-y-2">
|
|
<p className="text-sm">API Key</p>
|
|
<div className="flex items-center space-x-1 -ml-1">
|
|
{anonKey?.tags?.split(',').map((x: any, i: number) => (
|
|
<code key={`${x}${i}`} className="text-xs">
|
|
{x}
|
|
</code>
|
|
))}
|
|
<code className="text-xs">{'public'}</code>
|
|
</div>
|
|
</div>
|
|
}
|
|
copy={canReadAPIKeys && isNotUpdatingJwtSecret}
|
|
reveal={anonKey?.tags !== 'anon' && canReadAPIKeys && isNotUpdatingJwtSecret}
|
|
value={
|
|
!canReadAPIKeys
|
|
? 'You need additional permissions to view API keys'
|
|
: jwtSecretUpdateStatus === JwtSecretUpdateStatus.Failed
|
|
? 'JWT secret update failed, new API key may have issues'
|
|
: jwtSecretUpdateStatus === JwtSecretUpdateStatus.Updating
|
|
? 'Updating JWT secret...'
|
|
: apiService?.defaultApiKey
|
|
}
|
|
onChange={() => {}}
|
|
descriptionText={
|
|
<p>
|
|
This key is safe to use in a browser if you have enabled Row Level Security (RLS)
|
|
for your tables and configured policies. You may also use the service key which
|
|
can be found{' '}
|
|
<Link
|
|
href={`/project/${projectRef}/settings/api`}
|
|
className="transition text-brand hover:text-brand-600"
|
|
>
|
|
here
|
|
</Link>{' '}
|
|
to bypass RLS.
|
|
</p>
|
|
}
|
|
/>
|
|
</Panel.Content>
|
|
<div className="border-t border-panel-border-interior-light dark:border-panel-border-interior-dark">
|
|
<div className="flex items-center bg-studio">
|
|
{availableLanguages.map((language) => {
|
|
const isSelected = selectedLanguage.key === language.key
|
|
return (
|
|
<div
|
|
key={language.key}
|
|
className={[
|
|
'px-3 py-1 text-sm cursor-pointer transition',
|
|
`${!isSelected ? 'bg-studio text-foreground-light' : 'bg-surface-100'}`,
|
|
].join(' ')}
|
|
onClick={() => setSelectedLanguage(language)}
|
|
>
|
|
{language.name}
|
|
</div>
|
|
)
|
|
})}
|
|
</div>
|
|
<div className="bg-surface-100 px-4 py-6 min-h-[200px]">
|
|
<SimpleCodeBlock className={selectedLanguage.key}>
|
|
{selectedLanguageSnippet}
|
|
</SimpleCodeBlock>
|
|
</div>
|
|
</div>
|
|
</>
|
|
)}
|
|
</Panel>
|
|
)
|
|
}
|
|
|
|
export default APIKeys
|