mirror of
https://github.com/supabase/supabase.git
synced 2026-05-21 12:57:15 +08:00
Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: Ivan Vasilov <vasilov.ivan@gmail.com>
487 lines
17 KiB
TypeScript
487 lines
17 KiB
TypeScript
/* Based on https://github.com/graphql/graphiql/blob/main/packages/graphiql/src/components/GraphiQL.tsx */
|
|
|
|
import {
|
|
ChevronDownIcon,
|
|
ChevronUpIcon,
|
|
CopyIcon,
|
|
ExecuteButton,
|
|
GraphiQLProvider,
|
|
HeaderEditor,
|
|
MergeIcon,
|
|
PlusIcon,
|
|
PrettifyIcon,
|
|
QueryEditor,
|
|
ReloadIcon,
|
|
ResponseEditor,
|
|
Spinner,
|
|
Tab,
|
|
Tabs,
|
|
ToolbarButton,
|
|
Tooltip,
|
|
UnStyledButton,
|
|
useCopyQuery,
|
|
useDragResize,
|
|
useEditorContext,
|
|
useExecutionContext,
|
|
useMergeQuery,
|
|
usePluginContext,
|
|
usePrettifyEditors,
|
|
useSchemaContext,
|
|
useTheme,
|
|
VariableEditor,
|
|
} from '@graphiql/react'
|
|
import { Fetcher } from '@graphiql/toolkit'
|
|
import { PermissionAction } from '@supabase/shared-types/out/constants'
|
|
import { LOCAL_STORAGE_KEYS } from 'common'
|
|
import { AlertTriangle, XIcon } from 'lucide-react'
|
|
import { MouseEventHandler, useCallback, useEffect, useState } from 'react'
|
|
import { Alert_Shadcn_, AlertDescription_Shadcn_, AlertTitle_Shadcn_, Button, cn } from 'ui'
|
|
|
|
import { RoleImpersonationSelector } from '../RoleImpersonationSelector'
|
|
import styles from './graphiql.module.css'
|
|
import { useAsyncCheckPermissions } from '@/hooks/misc/useCheckPermissions'
|
|
import { useLocalStorage } from '@/hooks/misc/useLocalStorage'
|
|
|
|
export interface GraphiQLProps {
|
|
fetcher: Fetcher
|
|
theme?: 'dark' | 'light'
|
|
}
|
|
|
|
export default function GraphiQL({ fetcher, theme = 'dark' }: GraphiQLProps) {
|
|
// Ensure props are correct
|
|
if (typeof fetcher !== 'function') {
|
|
throw new TypeError(
|
|
'The `GraphiQL` component requires a `fetcher` function to be passed as prop.'
|
|
)
|
|
}
|
|
|
|
return (
|
|
<GraphiQLProvider fetcher={fetcher}>
|
|
<GraphiQLInterface theme={theme} />
|
|
</GraphiQLProvider>
|
|
)
|
|
}
|
|
|
|
interface GraphiQLInterfaceProps {
|
|
theme: 'dark' | 'light'
|
|
}
|
|
|
|
const GraphiQLInterface = ({ theme }: GraphiQLInterfaceProps) => {
|
|
const editorContext = useEditorContext({ nonNull: true })
|
|
const executionContext = useExecutionContext({ nonNull: true })
|
|
const schemaContext = useSchemaContext({ nonNull: true })
|
|
const pluginContext = usePluginContext()
|
|
|
|
const copy = useCopyQuery()
|
|
const merge = useMergeQuery()
|
|
const prettify = usePrettifyEditors()
|
|
|
|
const { can: canReadJWTSecret } = useAsyncCheckPermissions(
|
|
PermissionAction.READ,
|
|
'field.jwt_secret'
|
|
)
|
|
|
|
const [rlsBypassedWarningDismissed, setRlsBypassedWarningDismissed] = useLocalStorage(
|
|
LOCAL_STORAGE_KEYS.GRAPHIQL_RLS_BYPASS_WARNING,
|
|
false
|
|
)
|
|
|
|
const { setTheme } = useTheme()
|
|
useEffect(() => {
|
|
setTheme(theme)
|
|
}, [theme])
|
|
|
|
const PluginContent = pluginContext?.visiblePlugin?.content
|
|
|
|
const pluginResize = useDragResize({
|
|
defaultSizeRelation: 1 / 3,
|
|
direction: 'horizontal',
|
|
initiallyHidden: pluginContext?.visiblePlugin ? undefined : 'second',
|
|
onHiddenElementChange: (resizableElement) => {
|
|
if (resizableElement === 'second') {
|
|
pluginContext?.setVisiblePlugin(null)
|
|
}
|
|
},
|
|
sizeThresholdSecond: 200,
|
|
storageKey: 'docExplorerFlex',
|
|
})
|
|
const editorResize = useDragResize({
|
|
direction: 'horizontal',
|
|
storageKey: 'editorFlex',
|
|
})
|
|
const editorToolsResize = useDragResize({
|
|
defaultSizeRelation: 3,
|
|
direction: 'vertical',
|
|
initiallyHidden: (() => {
|
|
return editorContext.initialVariables || editorContext.initialHeaders ? undefined : 'second'
|
|
})(),
|
|
sizeThresholdSecond: 60,
|
|
storageKey: 'secondaryEditorFlex',
|
|
})
|
|
|
|
const [activeSecondaryEditor, setActiveSecondaryEditor] = useState<
|
|
'variables' | 'headers' | 'role-impersonation'
|
|
>(() => {
|
|
return !editorContext.initialVariables && editorContext.initialHeaders ? 'headers' : 'variables'
|
|
})
|
|
|
|
const toolbar = (
|
|
<>
|
|
<ToolbarButton onClick={prettify} label="Prettify query (Shift-Ctrl-P)">
|
|
<PrettifyIcon className="graphiql-toolbar-icon" aria-hidden="true" />
|
|
</ToolbarButton>
|
|
<ToolbarButton onClick={merge} label="Merge fragments into query (Shift-Ctrl-M)">
|
|
<MergeIcon className="graphiql-toolbar-icon" aria-hidden="true" />
|
|
</ToolbarButton>
|
|
<ToolbarButton onClick={copy} label="Copy query (Shift-Ctrl-C)">
|
|
<CopyIcon className="graphiql-toolbar-icon" aria-hidden="true" />
|
|
</ToolbarButton>
|
|
</>
|
|
)
|
|
|
|
const onClickReference = useCallback(() => {
|
|
if (pluginResize.hiddenElement === 'second') {
|
|
pluginResize.setHiddenElement(null)
|
|
}
|
|
}, [pluginResize])
|
|
|
|
const handleAddTab = editorContext.addTab
|
|
const handleRefetchSchema = schemaContext.introspect
|
|
const handleReorder = editorContext.moveTab
|
|
|
|
const handlePluginClick: MouseEventHandler<HTMLButtonElement> = useCallback(
|
|
(e) => {
|
|
const context = pluginContext!
|
|
const pluginIndex = Number(e.currentTarget.dataset.index!)
|
|
const plugin = context.plugins.find((_, index) => pluginIndex === index)!
|
|
const isVisible = plugin === context.visiblePlugin
|
|
if (isVisible) {
|
|
context.setVisiblePlugin(null)
|
|
pluginResize.setHiddenElement('second')
|
|
} else {
|
|
context.setVisiblePlugin(plugin)
|
|
pluginResize.setHiddenElement(null)
|
|
}
|
|
},
|
|
[pluginContext, pluginResize]
|
|
)
|
|
|
|
const handleToolsTabClick: MouseEventHandler<HTMLButtonElement> = useCallback(
|
|
(event) => {
|
|
if (editorToolsResize.hiddenElement === 'second') {
|
|
editorToolsResize.setHiddenElement(null)
|
|
}
|
|
setActiveSecondaryEditor(
|
|
event.currentTarget.dataset.name as 'variables' | 'headers' | 'role-impersonation'
|
|
)
|
|
},
|
|
[editorToolsResize]
|
|
)
|
|
|
|
const toggleEditorTools: MouseEventHandler<HTMLButtonElement> = useCallback(() => {
|
|
editorToolsResize.setHiddenElement(
|
|
editorToolsResize.hiddenElement === 'second' ? null : 'second'
|
|
)
|
|
}, [editorToolsResize])
|
|
|
|
const addTab = (
|
|
<Tooltip label="Add tab">
|
|
<UnStyledButton
|
|
type="button"
|
|
className="graphiql-tab-add text-sm"
|
|
onClick={handleAddTab}
|
|
aria-label="Add tab"
|
|
>
|
|
<PlusIcon aria-hidden="true" />
|
|
</UnStyledButton>
|
|
</Tooltip>
|
|
)
|
|
|
|
const hasSingleTab = editorContext.tabs.length === 1
|
|
|
|
return (
|
|
<Tooltip.Provider>
|
|
<div className={cn('graphiql-container', styles.graphiqlContainer)}>
|
|
<div className="graphiql-main">
|
|
<div
|
|
ref={pluginResize.firstRef}
|
|
style={{ minWidth: '750px' }}
|
|
className={cn('graphiql-sessions', styles.graphiqlSessions)}
|
|
>
|
|
<div
|
|
className={cn(
|
|
'graphiql-session-header',
|
|
!hasSingleTab && styles.graphiqlSessionHeader
|
|
)}
|
|
>
|
|
<Tabs
|
|
values={editorContext.tabs}
|
|
onReorder={handleReorder}
|
|
aria-label="Select active operation"
|
|
>
|
|
{editorContext.tabs.length > 1 && (
|
|
<>
|
|
{editorContext.tabs.map((tab, index) => (
|
|
<Tab
|
|
key={tab.id}
|
|
value={tab}
|
|
isActive={index === editorContext.activeTabIndex}
|
|
>
|
|
<Tab.Button
|
|
aria-controls="graphiql-session"
|
|
id={`graphiql-session-tab-${index}`}
|
|
onClick={() => {
|
|
executionContext.stop()
|
|
editorContext.changeTab(index)
|
|
}}
|
|
>
|
|
{tab.title}
|
|
</Tab.Button>
|
|
<Tab.Close
|
|
onClick={() => {
|
|
if (editorContext.activeTabIndex === index) {
|
|
executionContext.stop()
|
|
}
|
|
editorContext.closeTab(index)
|
|
}}
|
|
/>
|
|
</Tab>
|
|
))}
|
|
{addTab}
|
|
</>
|
|
)}
|
|
</Tabs>
|
|
<div className="graphiql-session-header-right">
|
|
{hasSingleTab && <div className={styles.graphiqlAddTabWrapper}>{addTab}</div>}
|
|
</div>
|
|
</div>
|
|
<div
|
|
role="tabpanel"
|
|
id="graphiql-session"
|
|
className={cn('graphiql-session', styles.graphiqlSession)}
|
|
aria-labelledby={`graphiql-session-tab-${editorContext.activeTabIndex}`}
|
|
>
|
|
<div ref={editorResize.firstRef}>
|
|
<div
|
|
className={cn(
|
|
'graphiql-editors',
|
|
styles.graphiqlEditors,
|
|
hasSingleTab && 'full-height'
|
|
)}
|
|
>
|
|
<div ref={editorToolsResize.firstRef}>
|
|
<section
|
|
className={cn('graphiql-query-editor text-sm', styles.graphiqlQueryEditor)}
|
|
aria-label="Query Editor"
|
|
>
|
|
<QueryEditor onClickReference={onClickReference} />
|
|
<div className="graphiql-toolbar" role="toolbar" aria-label="Editor Commands">
|
|
<ExecuteButton />
|
|
{toolbar}
|
|
</div>
|
|
</section>
|
|
</div>
|
|
|
|
<div ref={editorToolsResize.dragBarRef}>
|
|
<div className="graphiql-editor-tools">
|
|
<UnStyledButton
|
|
type="button"
|
|
className={
|
|
activeSecondaryEditor === 'variables' &&
|
|
editorToolsResize.hiddenElement !== 'second'
|
|
? 'active text-sm'
|
|
: 'text-sm'
|
|
}
|
|
onClick={handleToolsTabClick}
|
|
data-name="variables"
|
|
>
|
|
Variables
|
|
</UnStyledButton>
|
|
|
|
<UnStyledButton
|
|
type="button"
|
|
className={
|
|
activeSecondaryEditor === 'headers' &&
|
|
editorToolsResize.hiddenElement !== 'second'
|
|
? 'active text-sm'
|
|
: 'text-sm'
|
|
}
|
|
onClick={handleToolsTabClick}
|
|
data-name="headers"
|
|
>
|
|
Headers
|
|
</UnStyledButton>
|
|
|
|
{canReadJWTSecret && (
|
|
<UnStyledButton
|
|
type="button"
|
|
className={
|
|
activeSecondaryEditor === 'role-impersonation' &&
|
|
editorToolsResize.hiddenElement !== 'second'
|
|
? 'active text-sm'
|
|
: 'text-sm'
|
|
}
|
|
onClick={handleToolsTabClick}
|
|
data-name="role-impersonation"
|
|
>
|
|
Role Impersonation
|
|
</UnStyledButton>
|
|
)}
|
|
|
|
<Tooltip
|
|
label={
|
|
editorToolsResize.hiddenElement === 'second'
|
|
? 'Show editor tools'
|
|
: 'Hide editor tools'
|
|
}
|
|
>
|
|
<UnStyledButton
|
|
type="button"
|
|
onClick={toggleEditorTools}
|
|
aria-label={
|
|
editorToolsResize.hiddenElement === 'second'
|
|
? 'Show editor tools'
|
|
: 'Hide editor tools'
|
|
}
|
|
className="graphiql-toggle-editor-tools text-sm"
|
|
>
|
|
{editorToolsResize.hiddenElement === 'second' ? (
|
|
<ChevronUpIcon className="graphiql-chevron-icon" aria-hidden="true" />
|
|
) : (
|
|
<ChevronDownIcon className="graphiql-chevron-icon" aria-hidden="true" />
|
|
)}
|
|
</UnStyledButton>
|
|
</Tooltip>
|
|
</div>
|
|
</div>
|
|
|
|
<div ref={editorToolsResize.secondRef}>
|
|
<section
|
|
className="graphiql-editor-tool text-sm"
|
|
aria-label={activeSecondaryEditor === 'variables' ? 'Variables' : 'Headers'}
|
|
>
|
|
<VariableEditor
|
|
isHidden={activeSecondaryEditor !== 'variables'}
|
|
onClickReference={onClickReference}
|
|
/>
|
|
|
|
<HeaderEditor isHidden={activeSecondaryEditor !== 'headers'} />
|
|
|
|
{canReadJWTSecret && (
|
|
<div
|
|
className={cn(
|
|
'graphiql-editor px-1',
|
|
activeSecondaryEditor !== 'role-impersonation' && 'hidden'
|
|
)}
|
|
>
|
|
<RoleImpersonationSelector padded={false} />
|
|
</div>
|
|
)}
|
|
</section>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div
|
|
className={cn('graphiql-horizontal-drag-bar', styles.graphiqlHorizontalDragBar)}
|
|
ref={editorResize.dragBarRef}
|
|
/>
|
|
|
|
<div ref={editorResize.secondRef}>
|
|
<div
|
|
className={cn(
|
|
'graphiql-response text-sm relative',
|
|
hasSingleTab
|
|
? styles.graphiqlResponseSingleTab
|
|
: styles.graphiqlResponseMultiTab
|
|
)}
|
|
>
|
|
{executionContext.isFetching ? <Spinner /> : null}
|
|
<ResponseEditor />
|
|
|
|
{!rlsBypassedWarningDismissed && (
|
|
<Alert_Shadcn_ variant="warning" className="absolute bottom-[5px] right-[5px]">
|
|
<AlertTriangle strokeWidth={2} />
|
|
<AlertTitle_Shadcn_ className="leading-5 text-foreground">
|
|
Please note that queries and mutations run in GraphiQL now use the service
|
|
role key by default.
|
|
<br />
|
|
<span className="text-amber-900">RLS will be bypassed.</span>
|
|
</AlertTitle_Shadcn_>
|
|
<AlertDescription_Shadcn_>
|
|
You can send queries as a specific role/user by using the role impersonation
|
|
tab.
|
|
</AlertDescription_Shadcn_>
|
|
<Button
|
|
type="outline"
|
|
aria-label="Dismiss"
|
|
className="absolute top-2 right-2 p-1 !pl-1"
|
|
onClick={() => {
|
|
setRlsBypassedWarningDismissed(true)
|
|
}}
|
|
>
|
|
<XIcon width={14} height={14} />
|
|
</Button>
|
|
</Alert_Shadcn_>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
{pluginContext?.visiblePlugin && (
|
|
<div className="graphiql-horizontal-drag-bar" ref={pluginResize.dragBarRef} />
|
|
)}
|
|
<div
|
|
ref={pluginResize.secondRef}
|
|
style={{
|
|
// Make sure the container shrinks when containing long
|
|
// non-breaking texts
|
|
minWidth: '200px',
|
|
}}
|
|
>
|
|
<div className="graphiql-plugin">{PluginContent ? <PluginContent /> : null}</div>
|
|
</div>
|
|
</div>
|
|
<div className={cn('graphiql-sidebar', styles.graphiqlSidebar)}>
|
|
<div className="graphiql-sidebar-section">
|
|
{pluginContext?.plugins.map((plugin, index) => {
|
|
const isVisible = plugin === pluginContext.visiblePlugin
|
|
const label = `${isVisible ? 'Hide' : 'Show'} ${plugin.title}`
|
|
const Icon = plugin.icon
|
|
return (
|
|
<Tooltip key={plugin.title} label={label}>
|
|
<UnStyledButton
|
|
type="button"
|
|
className={isVisible ? 'active text-sm' : 'text-sm'}
|
|
onClick={handlePluginClick}
|
|
data-index={index}
|
|
aria-label={label}
|
|
>
|
|
<Icon aria-hidden="true" />
|
|
</UnStyledButton>
|
|
</Tooltip>
|
|
)
|
|
})}
|
|
</div>
|
|
<div className="graphiql-sidebar-section">
|
|
<Tooltip label="Re-fetch GraphQL schema">
|
|
<UnStyledButton
|
|
type="button"
|
|
className="text-sm"
|
|
disabled={schemaContext.isFetching}
|
|
onClick={handleRefetchSchema}
|
|
aria-label="Re-fetch GraphQL schema"
|
|
>
|
|
<ReloadIcon
|
|
className={schemaContext.isFetching ? 'graphiql-spin' : ''}
|
|
aria-hidden="true"
|
|
/>
|
|
</UnStyledButton>
|
|
</Tooltip>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</Tooltip.Provider>
|
|
)
|
|
}
|