mirror of
https://github.com/langgenius/dify.git
synced 2026-05-01 01:00:51 -04:00
Co-authored-by: CodingOnStar <hanxujiang@dify.com> Co-authored-by: autofix-ci[bot] <114827586+autofix-ci[bot]@users.noreply.github.com>
218 lines
7.0 KiB
TypeScript
218 lines
7.0 KiB
TypeScript
import type { PluginPayload } from '../types'
|
|
import type {
|
|
FormRefObject,
|
|
FormSchema,
|
|
} from '@/app/components/base/form/types'
|
|
import { Button } from '@langgenius/dify-ui/button'
|
|
import { Dialog, DialogCloseButton, DialogContent, DialogTitle } from '@langgenius/dify-ui/dialog'
|
|
import { toast } from '@langgenius/dify-ui/toast'
|
|
import {
|
|
useForm,
|
|
useStore,
|
|
} from '@tanstack/react-form'
|
|
import {
|
|
memo,
|
|
useCallback,
|
|
useRef,
|
|
useState,
|
|
} from 'react'
|
|
import { useTranslation } from 'react-i18next'
|
|
import AuthForm from '@/app/components/base/form/form-scenarios/auth'
|
|
import { ReadmeEntrance } from '../../readme-panel/entrance'
|
|
import { ReadmeShowType } from '../../readme-panel/store'
|
|
import {
|
|
useDeletePluginOAuthCustomClientHook,
|
|
useInvalidPluginOAuthClientSchemaHook,
|
|
useSetPluginOAuthCustomClientHook,
|
|
} from '../hooks/use-credential'
|
|
|
|
export type OAuthClientSettingsProps = {
|
|
pluginPayload: PluginPayload
|
|
open?: boolean
|
|
onOpenChange?: (open: boolean) => void
|
|
onClose?: () => void
|
|
editValues?: Record<string, unknown>
|
|
disabled?: boolean
|
|
schemas: FormSchema[]
|
|
onAuth?: () => Promise<void>
|
|
hasOriginalClientParams?: boolean
|
|
onUpdate?: () => void
|
|
}
|
|
const OAuthClientSettings = ({
|
|
pluginPayload,
|
|
open = true,
|
|
onOpenChange,
|
|
onClose,
|
|
editValues,
|
|
disabled,
|
|
schemas,
|
|
onAuth,
|
|
hasOriginalClientParams,
|
|
onUpdate,
|
|
}: OAuthClientSettingsProps) => {
|
|
const { t } = useTranslation()
|
|
const [doingAction, setDoingAction] = useState(false)
|
|
const doingActionRef = useRef(doingAction)
|
|
const handleSetDoingAction = useCallback((value: boolean) => {
|
|
doingActionRef.current = value
|
|
setDoingAction(value)
|
|
}, [])
|
|
const handleOpenChange = useCallback((nextOpen: boolean) => {
|
|
onOpenChange?.(nextOpen)
|
|
if (!nextOpen)
|
|
onClose?.()
|
|
}, [onClose, onOpenChange])
|
|
const defaultValues = schemas.reduce((acc, schema) => {
|
|
if (schema.default)
|
|
acc[schema.name] = schema.default
|
|
return acc
|
|
}, {} as Record<string, unknown>)
|
|
const { mutateAsync: setPluginOAuthCustomClient } = useSetPluginOAuthCustomClientHook(pluginPayload)
|
|
const invalidPluginOAuthClientSchema = useInvalidPluginOAuthClientSchemaHook(pluginPayload)
|
|
const formRef = useRef<FormRefObject>(null)
|
|
const handleConfirm = useCallback(async () => {
|
|
if (doingActionRef.current)
|
|
return
|
|
|
|
try {
|
|
const {
|
|
isCheckValidated,
|
|
values,
|
|
} = formRef.current?.getFormValues({
|
|
needCheckValidatedValues: true,
|
|
needTransformWhenSecretFieldIsPristine: true,
|
|
}) || { isCheckValidated: false, values: {} }
|
|
if (!isCheckValidated)
|
|
throw new Error('error')
|
|
const {
|
|
__oauth_client__,
|
|
...restValues
|
|
} = values
|
|
|
|
handleSetDoingAction(true)
|
|
await setPluginOAuthCustomClient({
|
|
client_params: restValues,
|
|
enable_oauth_custom_client: __oauth_client__ === 'custom',
|
|
})
|
|
toast.success(t('api.actionSuccess', { ns: 'common' }))
|
|
|
|
onOpenChange?.(false)
|
|
onClose?.()
|
|
onUpdate?.()
|
|
invalidPluginOAuthClientSchema()
|
|
}
|
|
finally {
|
|
handleSetDoingAction(false)
|
|
}
|
|
}, [onClose, onOpenChange, onUpdate, invalidPluginOAuthClientSchema, setPluginOAuthCustomClient, t, handleSetDoingAction])
|
|
|
|
const handleConfirmAndAuthorize = useCallback(async () => {
|
|
await handleConfirm()
|
|
if (onAuth)
|
|
await onAuth()
|
|
}, [handleConfirm, onAuth])
|
|
const { mutateAsync: deletePluginOAuthCustomClient } = useDeletePluginOAuthCustomClientHook(pluginPayload)
|
|
const handleRemove = useCallback(async () => {
|
|
if (doingActionRef.current)
|
|
return
|
|
|
|
try {
|
|
handleSetDoingAction(true)
|
|
await deletePluginOAuthCustomClient()
|
|
toast.success(t('api.actionSuccess', { ns: 'common' }))
|
|
onOpenChange?.(false)
|
|
onClose?.()
|
|
onUpdate?.()
|
|
invalidPluginOAuthClientSchema()
|
|
}
|
|
finally {
|
|
handleSetDoingAction(false)
|
|
}
|
|
}, [onUpdate, invalidPluginOAuthClientSchema, deletePluginOAuthCustomClient, t, handleSetDoingAction, onClose, onOpenChange])
|
|
const form = useForm({
|
|
defaultValues: editValues || defaultValues,
|
|
})
|
|
const __oauth_client__ = useStore(form.store, s => s.values.__oauth_client__)
|
|
const isDisabled = disabled || doingAction
|
|
|
|
return (
|
|
<Dialog
|
|
open={open}
|
|
onOpenChange={handleOpenChange}
|
|
>
|
|
<DialogContent
|
|
backdropProps={{ forceRender: true }}
|
|
className="w-[480px]! max-w-[calc(100vw-2rem)]! p-0!"
|
|
>
|
|
<div data-testid="modal" className="flex max-h-[80dvh] flex-col">
|
|
<div className="relative shrink-0 p-6 pr-14 pb-3">
|
|
<DialogTitle data-testid="modal-title" className="title-2xl-semi-bold text-text-primary">
|
|
{t('auth.oauthClientSettings', { ns: 'plugin' })}
|
|
</DialogTitle>
|
|
<DialogCloseButton
|
|
data-testid="modal-x-close"
|
|
className="top-5 right-5 h-8 w-8 rounded-lg"
|
|
/>
|
|
</div>
|
|
<div className="min-h-0 flex-1 overflow-y-auto px-6 py-3 pt-0">
|
|
{pluginPayload.detail && (
|
|
<ReadmeEntrance pluginDetail={pluginPayload.detail} showType={ReadmeShowType.modal} />
|
|
)}
|
|
<AuthForm
|
|
formFromProps={form}
|
|
ref={formRef}
|
|
formSchemas={schemas}
|
|
defaultValues={editValues || defaultValues}
|
|
disabled={disabled}
|
|
/>
|
|
</div>
|
|
<div className="flex shrink-0 justify-between p-6 pt-5">
|
|
<div>
|
|
{__oauth_client__ === 'custom' && hasOriginalClientParams && (
|
|
<Button
|
|
data-testid="modal-extra"
|
|
variant="secondary"
|
|
className="text-components-button-destructive-secondary-text"
|
|
disabled={isDisabled || !editValues}
|
|
onClick={handleRemove}
|
|
>
|
|
{t('operation.remove', { ns: 'common' })}
|
|
</Button>
|
|
)}
|
|
</div>
|
|
<div className="flex items-center">
|
|
<Button
|
|
data-testid="modal-close"
|
|
variant="secondary"
|
|
onClick={() => handleOpenChange(false)}
|
|
disabled={isDisabled}
|
|
>
|
|
{t('operation.cancel', { ns: 'common' })}
|
|
</Button>
|
|
<div className="mx-3 h-4 w-px bg-divider-regular"></div>
|
|
<Button
|
|
data-testid="modal-cancel"
|
|
onClick={handleConfirm}
|
|
disabled={isDisabled}
|
|
>
|
|
{t('auth.saveOnly', { ns: 'plugin' })}
|
|
</Button>
|
|
<Button
|
|
data-testid="modal-confirm"
|
|
className="ml-2"
|
|
variant="primary"
|
|
onClick={handleConfirmAndAuthorize}
|
|
disabled={isDisabled}
|
|
>
|
|
{t('auth.saveAndAuth', { ns: 'plugin' })}
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</DialogContent>
|
|
</Dialog>
|
|
)
|
|
}
|
|
|
|
export default memo(OAuthClientSettings)
|