Files
supabase/apps/studio/components/interfaces/Database/Functions/CreateFunction.tsx
Jonathan Summers-Muir c4a492002b Chore/move confirmation dialogs (#21651)
* move confirmation dialogs

* Update index.tsx

* Update ExtensionCard.tsx
2024-03-01 18:16:25 +08:00

892 lines
26 KiB
TypeScript

import { filter, has, isEmpty, isNull, isUndefined, keyBy, mapValues, partition } from 'lodash'
import { makeAutoObservable } from 'mobx'
import { observer, useLocalObservable } from 'mobx-react-lite'
import { FormEvent, createContext, useContext, useEffect, useState } from 'react'
import toast from 'react-hot-toast'
import { Button, IconPlus, IconTrash, Input, Listbox, Modal, Radio, SidePanel, Toggle } from 'ui'
import { POSTGRES_DATA_TYPES } from 'components/interfaces/TableGridEditor/SidePanelEditor/SidePanelEditor.constants'
import { useProjectContext } from 'components/layouts/ProjectLayout/ProjectContext'
import ConfirmationModal from 'ui-patterns/Dialogs/ConfirmationModal'
import Panel from 'components/ui/Panel'
import SqlEditor from 'components/ui/SqlEditor'
import { useDatabaseExtensionsQuery } from 'data/database-extensions/database-extensions-query'
import { useDatabaseFunctionCreateMutation } from 'data/database-functions/database-functions-create-mutation'
import { useDatabaseFunctionUpdateMutation } from 'data/database-functions/database-functions-update-mutation'
import { useSchemasQuery } from 'data/database/schemas-query'
import { EXCLUDED_SCHEMAS } from 'lib/constants/schemas'
import { Dictionary } from 'types'
import { convertArgumentTypes, convertConfigParams, hasWhitespace } from './Functions.utils'
// [Refactor] Remove local state, just use the Form component
class CreateFunctionFormState {
id: number | undefined
originalName: string | undefined
// @ts-ignore
args: { value: { name: string; type: string; error?: string }[] }
// @ts-ignore
behavior: { value: string; error?: string }
// @ts-ignore
configParams: {
value: { name: string; value: string; error?: { name?: string; value?: string } }[]
}
// @ts-ignore
definition: { value: string; error?: string }
// @ts-ignore
language: { value: string; error?: string }
// @ts-ignore
name: { value: string; error?: string }
// @ts-ignore
returnType: { value: string; error?: string }
// @ts-ignore
schema: { value: string; error?: string }
// @ts-ignore
securityDefiner: { value: boolean; error?: string }
constructor() {
makeAutoObservable(this)
this.reset()
}
get requestBody() {
return {
id: this.id,
name: this.name.value,
schema: this.schema.value,
definition: this.definition.value,
return_type: this.returnType.value,
language: this.language.value,
behavior: this.behavior.value as 'VOLATILE' | 'STABLE' | 'IMMUTABLE',
security_definer: this.securityDefiner.value,
args: this.args.value.map((x: any) => `${x.name} ${x.type}`),
config_params: mapValues(keyBy(this.configParams.value, 'name'), 'value'),
}
}
reset(func?: Dictionary<any>) {
this.id = func?.id
this.originalName = func?.name
this.args = convertArgumentTypes(func?.argument_types)
this.behavior = { value: func?.behavior ?? 'VOLATILE' }
this.configParams = convertConfigParams(func?.config_params)
this.definition = { value: func?.definition ?? '' }
this.language = { value: func?.language ?? 'plpgsql' }
this.name = { value: func?.name ?? '' }
this.returnType = { value: func?.return_type ?? 'void' }
this.schema = { value: func?.schema ?? 'public' }
this.securityDefiner = { value: func?.security_definer ?? false }
}
update(state: Dictionary<any>) {
this.args = state.args
this.behavior = state.behavior
this.configParams = state.configParams
this.definition = state.definition
this.language = state.language
this.name = state.name
this.returnType = state.returnType
this.schema = state.schema
this.securityDefiner = state.securityDefiner
}
}
interface ICreateFunctionStore {
formState: CreateFunctionFormState
meta: any
schemas: Dictionary<any>[]
isEditing: boolean
onFormChange: (value: { key: string; value: any }) => void
onFormArrayChange: (value: {
operation: 'add' | 'delete' | 'update'
key: string
idx?: number
value?: any
}) => void
setSchemas: (value: Dictionary<any>[]) => void
validateForm: () => boolean
}
class CreateFunctionStore implements ICreateFunctionStore {
formState = new CreateFunctionFormState()
meta = null
schemas = []
advancedVisible = false
isDirty = false
constructor() {
makeAutoObservable(this)
}
get title() {
return this.formState.id
? `Edit '${this.formState.originalName}' function`
: 'Add a new function'
}
get isEditing() {
return this.formState.id != undefined
}
toggleAdvancedVisible = () => {
this.advancedVisible = !this.advancedVisible
}
setSchemas = (value: Dictionary<any>[]) => {
this.schemas = value as any
}
setIsDirty = (value: boolean) => {
this.isDirty = value
}
onFormChange = ({ key, value }: { key: string; value: any }) => {
this.isDirty = true
if (has(this.formState, key)) {
const temp = (this.formState as any)[key] as any
;(this.formState as any)[key] = { ...temp, value, error: undefined }
} else {
;(this.formState as any)[key] = { value }
}
}
onFormArrayChange = ({
operation,
key,
idx,
value,
}: {
operation: 'add' | 'delete' | 'update'
key: string
idx?: number
value?: any
}) => {
switch (operation) {
case 'add': {
if (has(this.formState, key)) {
// @ts-ignore
this.formState[key].value.push(value)
} else {
const values = [value]
// @ts-ignore
this.formState[key] = { value: [value] }
}
break
}
case 'delete': {
if (has(this.formState, key)) {
const temp = filter(
// @ts-ignore
this.formState[key].value,
(_: any, index: number) => index != idx
) as any
// @ts-ignore
this.formState[key].value = temp
}
break
}
default: {
if (has(this.formState, key) && !isUndefined(idx)) {
// @ts-ignore
this.formState[key].value[idx] = value
} else {
// @ts-ignore
this.formState[key] = { value: [value] }
}
}
}
}
validateForm = () => {
let isValidated = true
const _state = mapValues(this.formState, (x: { value: any }, key: string) => {
switch (key) {
case 'name': {
if (isEmpty(x.value) || hasWhitespace(x.value)) {
isValidated = false
return { ...x, error: 'Invalid function name' }
} else {
return x
}
}
case 'args': {
const temp = x.value?.map((i: Dictionary<any>) => {
if (isEmpty(i.name) || hasWhitespace(i.name)) {
isValidated = false
return { ...i, error: 'Invalid argument name' }
} else {
return i
}
})
x.value = temp
return x
}
case 'configParams': {
const temp = x.value?.map((i: Dictionary<any>) => {
const error: any = { name: undefined, value: undefined }
if (isEmpty(i.name) || hasWhitespace(i.name)) {
isValidated = false
error.name = 'Invalid config name'
}
if (isEmpty(i.value)) {
isValidated = false
error.value = 'Missing config value'
}
return { ...i, error }
})
x.value = temp
return x
}
default:
return x
}
})
if (!isValidated) {
this.formState.update(_state)
}
return isValidated
}
}
const CreateFunctionContext = createContext<ICreateFunctionStore | null>(null)
interface CreateFunctionProps {
func: any
visible: boolean
setVisible: (value: boolean) => void
}
const CreateFunction = ({ func, visible, setVisible }: CreateFunctionProps) => {
const { project } = useProjectContext()
const _localState = useLocalObservable(() => new CreateFunctionStore())
const [isClosingPanel, setIsClosingPanel] = useState<boolean>(false)
useSchemasQuery(
{
projectRef: project?.ref,
connectionString: project?.connectionString,
},
{
onSuccess(schemas) {
_localState.setSchemas(schemas)
},
}
)
const { mutate: createDatabaseFunction, isLoading: isCreating } =
useDatabaseFunctionCreateMutation()
const { mutate: updateDatabaseFunction, isLoading: isUpdating } =
useDatabaseFunctionUpdateMutation()
useEffect(() => {
_localState.formState.reset(func)
_localState.setIsDirty(false)
}, [visible, func])
async function handleSubmit() {
if (!project) return console.error('Project is required')
if (_localState.validateForm()) {
const body = _localState.formState.requestBody
if (body.id) {
updateDatabaseFunction(
{
id: body.id,
projectRef: project.ref,
connectionString: project.connectionString,
payload: body as any,
},
{
onSuccess: () => {
toast.success(`Successfully updated function ${body.name}`)
setVisible(!visible)
},
}
)
} else {
createDatabaseFunction(
{
projectRef: project.ref,
connectionString: project.connectionString,
payload: body,
},
{
onSuccess: () => {
toast.success(`Successfully created function ${body.name}`)
setVisible(!visible)
},
}
)
}
}
}
function isClosingSidePanel() {
_localState.isDirty ? setIsClosingPanel(true) : setVisible(!visible)
}
return (
<>
<SidePanel
size="large"
visible={visible}
onCancel={isClosingSidePanel}
header={_localState.title}
className="hooks-sidepanel mr-0 transform transition-all duration-300 ease-in-out"
loading={isCreating || isUpdating}
onConfirm={handleSubmit}
>
<CreateFunctionContext.Provider value={_localState}>
<div className="mt-4 space-y-10">
{_localState.isEditing ? (
<>
<SidePanel.Content>
<div className="space-y-4">
<InputName />
<SelectSchema />
</div>
</SidePanel.Content>
<SidePanel.Separator />
<SidePanel.Content>
<InputMultiArguments readonly={true} />
</SidePanel.Content>
<SidePanel.Separator />
<SidePanel.Content>
<InputDefinition />
</SidePanel.Content>
</>
) : (
<div className="space-y-6">
<SidePanel.Content>
<InputName />
</SidePanel.Content>
<SidePanel.Separator />
<SidePanel.Content>
<div className="space-y-4">
<SelectSchema />
<SelectReturnType />
</div>
</SidePanel.Content>
<SidePanel.Separator />
<SidePanel.Content>
<InputMultiArguments />
</SidePanel.Content>
<SidePanel.Separator />
<SidePanel.Content>
<InputDefinition />
</SidePanel.Content>
<SidePanel.Separator />
<SidePanel.Content>
<Panel>
<div className={`space-y-8 rounded bg-studio py-4`}>
<div className={`px-6`}>
<Toggle
onChange={() => _localState.toggleAdvancedVisible()}
label="Show advanced settings"
checked={_localState.advancedVisible}
labelOptional="These are settings that might be familiar for postgres heavy users "
/>
</div>
{/* advanced selections */}
</div>
</Panel>
</SidePanel.Content>
{_localState.advancedVisible && (
<>
<SidePanel.Content>
<div className="space-y-2">
<SelectLanguage />
<SelectBehavior />
</div>
</SidePanel.Content>
<SidePanel.Separator />
<SidePanel.Content>
<InputMultiConfigParams />
</SidePanel.Content>
<SidePanel.Separator />
<SidePanel.Content>
<RadioSecurity />
</SidePanel.Content>
</>
)}
</div>
)}
</div>
</CreateFunctionContext.Provider>
<ConfirmationModal
visible={isClosingPanel}
header="Discard changes"
buttonLabel="Discard"
onSelectCancel={() => setIsClosingPanel(false)}
onSelectConfirm={() => {
setIsClosingPanel(false)
setVisible(!visible)
}}
>
<Modal.Content>
<p className="py-4 text-sm text-foreground-light">
There are unsaved changes. Are you sure you want to close the panel? Your changes will
be lost.
</p>
</Modal.Content>
</ConfirmationModal>
</SidePanel>
</>
)
}
export default observer(CreateFunction)
const InputName = observer(({}) => {
const _localState = useContext(CreateFunctionContext)
return (
<Input
id="name"
label="Name of function"
layout="horizontal"
placeholder="Name of function"
value={_localState!.formState.name.value}
onChange={(e) =>
_localState!.onFormChange({
key: 'name',
value: e.target.value,
})
}
size="small"
error={_localState!.formState.name.error}
descriptionText="Name will also be used for the function name in postgres"
/>
)
})
interface InputMultiArgumentsProps {
readonly?: boolean
}
const InputMultiArguments = observer(({ readonly }: InputMultiArgumentsProps) => {
const _localState = useContext(CreateFunctionContext)
function onAddArgument() {
_localState!.onFormArrayChange({
key: 'args',
value: { name: '', type: 'bool' },
operation: 'add',
})
}
return (
<div>
<div className="flex flex-col">
<h5 className="text-base text-foreground">Arguments</h5>
<p className="text-sm text-foreground-light">
Arguments can be referenced in the function body using either names or numbers.
</p>
</div>
<div className="space-y-2 pt-4">
{readonly && isEmpty(_localState!.formState.args.value) && (
<span className="text-foreground-lighter">No argument for this function</span>
)}
{_localState!.formState.args.value.map(
(x: { name: string; type: string; error?: string }, idx: number) => (
<InputArgument
key={`arg-${idx}`}
idx={idx}
name={x.name}
type={x.type}
error={x.error}
readonly={readonly}
/>
)
)}
{!readonly && (
<div>
<Button type="default" icon={<IconPlus />} onClick={onAddArgument} disabled={readonly}>
Add a new argument
</Button>
</div>
)}
</div>
</div>
)
})
interface InputArgumentProps {
idx: number
name: string
type: string
error?: string
readonly?: boolean
}
const InputArgument = observer(({ idx, name, type, error, readonly }: InputArgumentProps) => {
const _localState = useContext(CreateFunctionContext)
function onNameChange(e: FormEvent<HTMLInputElement>) {
const _value = e.currentTarget.value
_localState!.onFormArrayChange({
key: 'args',
value: { name: _value, type },
idx,
operation: 'update',
})
}
function onTypeChange(_value: string) {
_localState!.onFormArrayChange({
key: 'args',
value: { name, type: _value },
idx,
operation: 'update',
})
}
function onDelete() {
_localState!.onFormArrayChange({
key: 'args',
idx,
operation: 'delete',
})
}
return (
<div className="flex flex-row space-x-1 items-center">
<Input
id={`name-${idx}`}
className="flex-1 flex-grow"
value={name}
placeholder="Name of argument"
onChange={onNameChange}
size="small"
error={error}
disabled={readonly}
/>
{readonly ? (
<Input disabled readOnly id={`type-${idx}`} size="small" value={type} className="flex-1" />
) : (
<Listbox
id={`type-${idx}`}
className="flex-1"
value={type}
size="small"
onChange={onTypeChange}
disabled={readonly}
>
<Listbox.Option value="integer" label="integer">
integer
</Listbox.Option>
{POSTGRES_DATA_TYPES.map((x: string) => (
<Listbox.Option key={x} value={x} label={x}>
{x}
</Listbox.Option>
))}
</Listbox>
)}
{!readonly && (
<Button type="danger" icon={<IconTrash size="tiny" />} onClick={onDelete} size="small" />
)}
</div>
)
})
const InputMultiConfigParams = observer(({}) => {
const _localState = useContext(CreateFunctionContext)
function onAddArgument() {
_localState!.onFormArrayChange({
key: 'configParams',
value: { name: '', value: '' },
operation: 'add',
})
}
return (
<div>
<div className="flex items-center justify-between">
<h5 className="text-base text-foreground">Config Params</h5>
</div>
<div className="space-y-2 pt-4">
{_localState!.formState.configParams.value.map(
(
x: { name: string; value: string; error?: { name?: string; value?: string } },
idx: number
) => (
<InputConfigParam
key={`configParam-${idx}`}
idx={idx}
name={x.name}
value={x.value}
error={x.error}
/>
)
)}
</div>
<div className="pt-2">
<Button type="default" icon={<IconPlus />} onClick={onAddArgument}>
Add a new config
</Button>
</div>
</div>
)
})
interface InputConfigParamProps {
idx: number
name: string
value: string
error?: { name?: string; value?: string }
}
const InputConfigParam = observer(({ idx, name, value, error }: InputConfigParamProps) => {
const _localState = useContext(CreateFunctionContext)
function onNameChange(e: FormEvent<HTMLInputElement>) {
const _value = e.currentTarget.value
_localState!.onFormArrayChange({
key: 'configParams',
value: { name: _value, value },
idx,
operation: 'update',
})
}
function onValueChange(e: FormEvent<HTMLInputElement>) {
const _value = e.currentTarget.value
_localState!.onFormArrayChange({
key: 'configParams',
value: { name, value: _value },
idx,
operation: 'update',
})
}
function onDelete() {
_localState!.onFormArrayChange({
key: 'configParams',
idx,
operation: 'delete',
})
}
return (
<div className="flex space-x-1 items-center">
<Input
id={`name-${idx}`}
className="flex-1"
placeholder="Name of config"
value={name}
onChange={onNameChange}
size="small"
error={error?.name}
/>
<Input
id={`value-${idx}`}
className="flex-1"
placeholder="Value of config"
value={value}
onChange={onValueChange}
size="small"
error={error?.value}
/>
<Button type="danger" icon={<IconTrash size="tiny" />} onClick={onDelete} size="small" />
</div>
)
})
const InputDefinition = observer(({}) => {
const _localState = useContext(CreateFunctionContext)
return (
<div className="space-y-4">
<div className="flex flex-col">
<h5 className="text-base text-foreground">Definition</h5>
<p className="text-sm text-foreground-light">
The language below should be written in `{_localState!.formState.language.value}`.
</p>
{!_localState?.isEditing && (
<p className="text-sm text-foreground-light">
Change the language in the Advanced Settings below.
</p>
)}
</div>
<div className="h-60 resize-y border border-default">
<SqlEditor
defaultValue={_localState!.formState.definition.value}
onInputChange={(value: string | undefined) => {
_localState!.onFormChange({
key: 'definition',
value: value,
})
}}
contextmenu={false}
/>
</div>
</div>
)
})
const SelectSchema = observer(({}) => {
const _localState = useContext(CreateFunctionContext)
const { project } = useProjectContext()
const { data } = useSchemasQuery({
projectRef: project?.ref,
connectionString: project?.connectionString,
})
const schemas = data?.filter((schema) => !EXCLUDED_SCHEMAS.includes(schema.name)) ?? []
return (
<Listbox
id="schema"
label="Schema"
size="small"
layout="horizontal"
placeholder="Pick a schema"
value={_localState!.formState.schema.value}
descriptionText="Tables made in the table editor will be in 'public'"
onChange={(value) => _localState!.onFormChange({ key: 'schema', value })}
>
{schemas.map((x) => (
<Listbox.Option key={x.name} label={x.name} value={x.name}>
{x.name}
</Listbox.Option>
))}
</Listbox>
)
})
const SelectLanguage = observer(({}) => {
const _localState = useContext(CreateFunctionContext)
const { project } = useProjectContext()
const { data } = useDatabaseExtensionsQuery({
projectRef: project?.ref,
connectionString: project?.connectionString,
})
const [enabledExtensions] = partition(data ?? [], (ext: any) => !isNull(ext.installed_version))
return (
<Listbox
id="language"
size="small"
label="Language"
layout="horizontal"
value={_localState!.formState.language.value}
placeholder="Pick a language"
onChange={(value) => _localState!.onFormChange({ key: 'language', value })}
>
<Listbox.Option value="sql" label="sql">
sql
</Listbox.Option>
{
//map through all selected extensions that start with pl
enabledExtensions
.filter((ex: any) => {
return ex.name.startsWith('pl')
})
.map((ex) => (
<Listbox.Option key={ex.name} value={ex.name} label={ex.name}>
{ex.name}
</Listbox.Option>
))
}
</Listbox>
)
})
const SelectReturnType = observer(({}) => {
const _localState = useContext(CreateFunctionContext)
return (
<Listbox
id="returnType"
size="small"
label="Return type"
layout="horizontal"
value={_localState!.formState.returnType.value}
onChange={(value) => _localState!.onFormChange({ key: 'returnType', value })}
>
<Listbox.Option value="void" label="void">
void
</Listbox.Option>
<Listbox.Option value="record" label="record">
record
</Listbox.Option>
<Listbox.Option value="trigger" label="trigger">
trigger
</Listbox.Option>
<Listbox.Option value="integer" label="integer">
integer
</Listbox.Option>
{POSTGRES_DATA_TYPES.map((x: string) => (
<Listbox.Option key={x} value={x} label={x}>
{x}
</Listbox.Option>
))}
</Listbox>
)
})
const SelectBehavior = observer(({}) => {
const _localState = useContext(CreateFunctionContext)
return (
<Listbox
id="behavior"
size="small"
label="Behavior"
layout="horizontal"
value={_localState!.formState.behavior.value}
onChange={(value) => _localState!.onFormChange({ key: 'behavior', value })}
>
<Listbox.Option value="IMMUTABLE" label="immutable">
immutable
</Listbox.Option>
<Listbox.Option value="STABLE" label="stable">
stable
</Listbox.Option>
<Listbox.Option value="VOLATILE" label="volatile">
volatile
</Listbox.Option>
</Listbox>
)
})
const RadioSecurity = observer(({}) => {
const _localState = useContext(CreateFunctionContext)
return (
<>
<div className="space-y-4">
<Radio.Group
type="cards"
label="Type of security"
layout="vertical"
onChange={(event) => {
_localState!.onFormChange({
key: 'securityDefiner',
value: event.target.value == 'SECURITY_DEFINER',
})
}}
value={
_localState!.formState.securityDefiner.value ? 'SECURITY_DEFINER' : 'SECURITY_INVOKER'
}
error={_localState!.formState.securityDefiner.error}
>
<Radio
id="SECURITY_INVOKER"
label="SECURITY INVOKER"
value="SECURITY_INVOKER"
checked={!_localState!.formState.securityDefiner.value}
description="Function is to be executed with the privileges of the user that calls it."
/>
<Radio
id="SECURITY_DEFINER"
label="SECURITY DEFINER"
value="SECURITY_DEFINER"
checked={_localState!.formState.securityDefiner.value}
description="Function is to be executed with the privileges of the user that created it."
/>
</Radio.Group>
</div>
</>
)
})