| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380 | import type { ChangeEvent, FC } from 'react'import { useState } from 'react'import useSWR from 'swr'import { useContext } from 'use-context-selector'import { useTranslation } from 'react-i18next'import { RiCloseLine } from '@remixicon/react'import ModerationContent from './moderation-content'import FormGeneration from './form-generation'import ApiBasedExtensionSelector from '@/app/components/header/account-setting/api-based-extension-page/selector'import Modal from '@/app/components/base/modal'import Button from '@/app/components/base/button'import Divider from '@/app/components/base/divider'import { BookOpen01 } from '@/app/components/base/icons/src/vender/line/education'import type { ModerationConfig, ModerationContentConfig } from '@/models/debug'import { useToastContext } from '@/app/components/base/toast'import {  fetchCodeBasedExtensionList,  fetchModelProviders,} from '@/service/common'import type { CodeBasedExtensionItem } from '@/models/common'import I18n from '@/context/i18n'import { LanguagesSupported } from '@/i18n/language'import { InfoCircle } from '@/app/components/base/icons/src/vender/line/general'import { useModalContext } from '@/context/modal-context'import { CustomConfigurationStatusEnum } from '@/app/components/header/account-setting/model-provider-page/declarations'import cn from '@/utils/classnames'const systemTypes = ['openai_moderation', 'keywords', 'api']type Provider = {  key: string  name: string  form_schema?: CodeBasedExtensionItem['form_schema']}type ModerationSettingModalProps = {  data: ModerationConfig  onCancel: () => void  onSave: (moderationConfig: ModerationConfig) => void}const ModerationSettingModal: FC<ModerationSettingModalProps> = ({  data,  onCancel,  onSave,}) => {  const { t } = useTranslation()  const { notify } = useToastContext()  const { locale } = useContext(I18n)  const { data: modelProviders, isLoading, mutate } = useSWR('/workspaces/current/model-providers', fetchModelProviders)  const [localeData, setLocaleData] = useState<ModerationConfig>(data)  const { setShowAccountSettingModal } = useModalContext()  const handleOpenSettingsModal = () => {    setShowAccountSettingModal({      payload: 'provider',      onCancelCallback: () => {        mutate()      },    })  }  const { data: codeBasedExtensionList } = useSWR(    '/code-based-extension?module=moderation',    fetchCodeBasedExtensionList,  )  const openaiProvider = modelProviders?.data.find(item => item.provider === 'langgenius/openai/openai')  const systemOpenaiProviderEnabled = openaiProvider?.system_configuration.enabled  const systemOpenaiProviderQuota = systemOpenaiProviderEnabled ? openaiProvider?.system_configuration.quota_configurations.find(item => item.quota_type === openaiProvider.system_configuration.current_quota_type) : undefined  const systemOpenaiProviderCanUse = systemOpenaiProviderQuota?.is_valid  const customOpenaiProvidersCanUse = openaiProvider?.custom_configuration.status === CustomConfigurationStatusEnum.active  const isOpenAIProviderConfigured = customOpenaiProvidersCanUse || systemOpenaiProviderCanUse  const providers: Provider[] = [    {      key: 'openai_moderation',      name: t('appDebug.feature.moderation.modal.provider.openai'),    },    {      key: 'keywords',      name: t('appDebug.feature.moderation.modal.provider.keywords'),    },    {      key: 'api',      name: t('common.apiBasedExtension.selector.title'),    },    ...(      codeBasedExtensionList        ? codeBasedExtensionList.data.map((item) => {          return {            key: item.name,            name: locale === 'zh-Hans' ? item.label['zh-Hans'] : item.label['en-US'],            form_schema: item.form_schema,          }        })        : []    ),  ]  const currentProvider = providers.find(provider => provider.key === localeData.type)  const handleDataTypeChange = (type: string) => {    let config: undefined | Record<string, any>    const currProvider = providers.find(provider => provider.key === type)    if (systemTypes.findIndex(t => t === type) < 0 && currProvider?.form_schema) {      config = currProvider?.form_schema.reduce((prev, next) => {        prev[next.variable] = next.default        return prev      }, {} as Record<string, any>)    }    setLocaleData({      ...localeData,      type,      config,    })  }  const handleDataKeywordsChange = (e: ChangeEvent<HTMLTextAreaElement>) => {    const value = e.target.value    const arr = value.split('\n').reduce((prev: string[], next: string) => {      if (next !== '')        prev.push(next.slice(0, 100))      if (next === '' && prev[prev.length - 1] !== '')        prev.push(next)      return prev    }, [])    setLocaleData({      ...localeData,      config: {        ...localeData.config,        keywords: arr.slice(0, 100).join('\n'),      },    })  }  const handleDataContentChange = (contentType: string, contentConfig: ModerationContentConfig) => {    setLocaleData({      ...localeData,      config: {        ...localeData.config,        [contentType]: contentConfig,      },    })  }  const handleDataApiBasedChange = (apiBasedExtensionId: string) => {    setLocaleData({      ...localeData,      config: {        ...localeData.config,        api_based_extension_id: apiBasedExtensionId,      },    })  }  const handleDataExtraChange = (extraValue: Record<string, string>) => {    setLocaleData({      ...localeData,      config: {        ...localeData.config,        ...extraValue,      },    })  }  const formatData = (originData: ModerationConfig) => {    const { enabled, type, config } = originData    const { inputs_config, outputs_config } = config!    const params: Record<string, string | undefined> = {}    if (type === 'keywords')      params.keywords = config?.keywords    if (type === 'api')      params.api_based_extension_id = config?.api_based_extension_id    if (systemTypes.findIndex(t => t === type) < 0 && currentProvider?.form_schema) {      currentProvider.form_schema.forEach((form) => {        params[form.variable] = config?.[form.variable]      })    }    return {      type,      enabled,      config: {        inputs_config: inputs_config || { enabled: false },        outputs_config: outputs_config || { enabled: false },        ...params,      },    }  }  const handleSave = () => {    if (localeData.type === 'openai_moderation' && !isOpenAIProviderConfigured)      return    if (!localeData.config?.inputs_config?.enabled && !localeData.config?.outputs_config?.enabled) {      notify({ type: 'error', message: t('appDebug.feature.moderation.modal.content.condition') })      return    }    if (localeData.type === 'keywords' && !localeData.config.keywords) {      notify({ type: 'error', message: t('appDebug.errorMessage.valueOfVarRequired', { key: locale !== LanguagesSupported[1] ? 'keywords' : '关键词' }) })      return    }    if (localeData.type === 'api' && !localeData.config.api_based_extension_id) {      notify({ type: 'error', message: t('appDebug.errorMessage.valueOfVarRequired', { key: locale !== LanguagesSupported[1] ? 'API Extension' : 'API 扩展' }) })      return    }    if (systemTypes.findIndex(t => t === localeData.type) < 0 && currentProvider?.form_schema) {      for (let i = 0; i < currentProvider.form_schema.length; i++) {        if (!localeData.config?.[currentProvider.form_schema[i].variable] && currentProvider.form_schema[i].required) {          notify({            type: 'error',            message: t('appDebug.errorMessage.valueOfVarRequired', { key: locale !== LanguagesSupported[1] ? currentProvider.form_schema[i].label['en-US'] : currentProvider.form_schema[i].label['zh-Hans'] }),          })          return        }      }    }    if (localeData.config.inputs_config?.enabled && !localeData.config.inputs_config.preset_response && localeData.type !== 'api') {      notify({ type: 'error', message: t('appDebug.feature.moderation.modal.content.errorMessage') })      return    }    if (localeData.config.outputs_config?.enabled && !localeData.config.outputs_config.preset_response && localeData.type !== 'api') {      notify({ type: 'error', message: t('appDebug.feature.moderation.modal.content.errorMessage') })      return    }    onSave(formatData(localeData))  }  return (    <Modal      isShow      onClose={() => { }}      className='!p-6 !mt-14 !max-w-none !w-[600px]'    >      <div className='flex items-center justify-between'>        <div className='text-text-primary title-2xl-semi-bold'>{t('appDebug.feature.moderation.modal.title')}</div>        <div className='p-1 cursor-pointer' onClick={onCancel}><RiCloseLine className='w-4 h-4 text-text-tertiary'/></div>      </div>      <div className='py-2'>        <div className='leading-9 text-sm font-medium text-text-primary'>          {t('appDebug.feature.moderation.modal.provider.title')}        </div>        <div className='grid gap-2.5 grid-cols-3'>          {            providers.map(provider => (              <div                key={provider.key}                className={cn(                  'flex items-center px-2 h-8 rounded-md system-sm-regular bg-components-option-card-option-bg border border-components-option-card-option-border text-text-secondary cursor-default',                  localeData.type !== provider.key && 'hover:bg-components-option-card-option-bg-hover hover:border-components-option-card-option-border-hover hover:shadow-xs cursor-pointer',                  localeData.type === provider.key && 'bg-components-option-card-option-selected-bg border-[1.5px] border-components-option-card-option-selected-border system-sm-medium shadow-xs',                  localeData.type === 'openai_moderation' && provider.key === 'openai_moderation' && !isOpenAIProviderConfigured && 'text-text-disabled',                )}                onClick={() => handleDataTypeChange(provider.key)}              >                <div className={cn(                  'mr-2 w-4 h-4 border border-components-radio-border bg-components-radio-bg shadow-xs rounded-full',                  localeData.type === provider.key && 'border-[5px] border-components-radio-border-checked',                )}></div>                {provider.name}              </div>            ))          }        </div>        {          !isLoading && !isOpenAIProviderConfigured && localeData.type === 'openai_moderation' && (            <div className='flex items-center mt-2 px-3 py-2 bg-[#FFFAEB] rounded-lg border border-[#FEF0C7]'>              <InfoCircle className='mr-1 w-4 h-4 text-[#F79009]' />              <div className='flex items-center text-xs font-medium text-gray-700'>                {t('appDebug.feature.moderation.modal.openaiNotConfig.before')}                <span                  className='text-primary-600 cursor-pointer'                  onClick={handleOpenSettingsModal}                >                   {t('common.settings.provider')}                 </span>                {t('appDebug.feature.moderation.modal.openaiNotConfig.after')}              </div>            </div>          )        }      </div>      {        localeData.type === 'keywords' && (          <div className='py-2'>            <div className='mb-1 text-sm font-medium text-text-primary'>{t('appDebug.feature.moderation.modal.provider.keywords')}</div>            <div className='mb-2 text-xs text-text-tertiary'>{t('appDebug.feature.moderation.modal.keywords.tip')}</div>            <div className='relative px-3 py-2 h-[88px] bg-components-input-bg-normal rounded-lg'>              <textarea                value={localeData.config?.keywords || ''}                onChange={handleDataKeywordsChange}                className='block w-full h-full bg-transparent text-sm text-text-secondary outline-none appearance-none resize-none'                placeholder={t('appDebug.feature.moderation.modal.keywords.placeholder') || ''}              />              <div className='absolute bottom-2 right-2 flex items-center px-1 h-5 rounded-md bg-background-section text-xs font-medium text-text-quaternary'>                <span>{(localeData.config?.keywords || '').split('\n').filter(Boolean).length}</span>/<span className='text-text-tertiary'>100 {t('appDebug.feature.moderation.modal.keywords.line')}</span>              </div>            </div>          </div>        )      }      {        localeData.type === 'api' && (          <div className='py-2'>            <div className='flex items-center justify-between h-9'>              <div className='text-sm font-medium text-text-primary'>{t('common.apiBasedExtension.selector.title')}</div>              <a                href={t('common.apiBasedExtension.linkUrl') || '/'}                target='_blank' rel='noopener noreferrer'                className='group flex items-center text-xs text-text-tertiary hover:text-primary-600'              >                <BookOpen01 className='mr-1 w-3 h-3 text-text-tertiary group-hover:text-primary-600' />                {t('common.apiBasedExtension.link')}              </a>            </div>            <ApiBasedExtensionSelector              value={localeData.config?.api_based_extension_id || ''}              onChange={handleDataApiBasedChange}            />          </div>        )      }      {        systemTypes.findIndex(t => t === localeData.type) < 0        && currentProvider?.form_schema        && (          <FormGeneration            forms={currentProvider?.form_schema}            value={localeData.config}            onChange={handleDataExtraChange}          />        )      }      <Divider bgStyle='gradient' className='my-3 h-px' />      <ModerationContent        title={t('appDebug.feature.moderation.modal.content.input') || ''}        config={localeData.config?.inputs_config || { enabled: false, preset_response: '' }}        onConfigChange={config => handleDataContentChange('inputs_config', config)}        info={(localeData.type === 'api' && t('appDebug.feature.moderation.modal.content.fromApi')) || ''}        showPreset={!(localeData.type === 'api')}      />      <ModerationContent        title={t('appDebug.feature.moderation.modal.content.output') || ''}        config={localeData.config?.outputs_config || { enabled: false, preset_response: '' }}        onConfigChange={config => handleDataContentChange('outputs_config', config)}        info={(localeData.type === 'api' && t('appDebug.feature.moderation.modal.content.fromApi')) || ''}        showPreset={!(localeData.type === 'api')}      />      <div className='mt-1 mb-8 text-xs font-medium text-text-tertiary'>{t('appDebug.feature.moderation.modal.content.condition')}</div>      <div className='flex items-center justify-end'>        <Button          onClick={onCancel}          className='mr-2'        >          {t('common.operation.cancel')}        </Button>        <Button          variant='primary'          onClick={handleSave}          disabled={localeData.type === 'openai_moderation' && !isOpenAIProviderConfigured}        >          {t('common.operation.save')}        </Button>      </div>    </Modal>  )}export default ModerationSettingModal
 |