fix: stabilize dynamic forms and mcp testing

This commit is contained in:
huanghuoguoguo
2026-05-17 10:25:08 +08:00
parent 31b2b6e6ca
commit 51d89caa91
6 changed files with 57 additions and 25 deletions

View File

@@ -262,6 +262,14 @@ class AgentRunnerRegistry:
stages = []
for descriptor in runners:
config_schema = []
for index, config_item in enumerate(descriptor.config_schema):
item = dict(config_item)
if not item.get('id'):
item_name = item.get('name') or str(index)
item['id'] = f'{descriptor.id}.{item_name}'
config_schema.append(item)
# Add runner option
options.append(
{
@@ -278,7 +286,7 @@ class AgentRunnerRegistry:
'name': descriptor.id,
'label': descriptor.label,
'description': descriptor.description,
'config': descriptor.config_schema,
'config': config_schema,
}
)

View File

@@ -502,12 +502,13 @@ export default function DynamicFormComponent({
}}
/>
{itemConfigList.map((config) => {
{itemConfigList.map((config, index) => {
// Create a normalized config with type converted to frontend format
const normalizedConfig = {
...config,
type: normalizeItemType(config.type),
};
const fieldKey = config.id || config.name || `field-${index}`;
if (config.show_if) {
const dependValue = resolveShowIfValue(
@@ -596,7 +597,7 @@ export default function DynamicFormComponent({
return (
<WebhookUrlField
key={config.id}
key={fieldKey}
label={extractI18nObject(config.label)}
description={
config.description
@@ -627,7 +628,7 @@ export default function DynamicFormComponent({
return (
<EmbedCodeField
key={config.id}
key={fieldKey}
label={extractI18nObject(config.label)}
description={
config.description
@@ -642,7 +643,7 @@ export default function DynamicFormComponent({
// QR code login button (e.g. Feishu one-click create, WeChat scan login)
if (config.type === 'qr-code-login') {
return (
<FormItem key={config.id}>
<FormItem key={fieldKey}>
<div
className="relative flex items-center gap-4 p-4 rounded-xl border-2 border-dashed cursor-pointer transition-all hover:border-solid hover:shadow-md group"
style={{
@@ -703,7 +704,7 @@ export default function DynamicFormComponent({
if (normalizedConfig.type === 'boolean') {
return (
<FormField
key={config.id}
key={fieldKey}
control={form.control}
name={config.name as keyof FormValues}
render={({ field }) => (
@@ -742,7 +743,7 @@ export default function DynamicFormComponent({
return (
<FormField
key={config.id}
key={fieldKey}
control={form.control}
name={config.name as keyof FormValues}
render={({ field }) => (

View File

@@ -253,6 +253,7 @@ export default function DynamicFormItemComponent({
type="number"
className="w-full max-w-xs"
{...field}
value={field.value ?? ''}
onChange={(e) => field.onChange(Number(e.target.value))}
/>
);
@@ -261,7 +262,11 @@ export default function DynamicFormItemComponent({
if (config.options && config.options.length > 0) {
return (
<div className="flex w-full max-w-md min-w-0 items-center gap-1.5">
<Input className="min-w-0 flex-1" {...field} />
<Input
className="min-w-0 flex-1"
{...field}
value={field.value ?? ''}
/>
<DropdownMenu>
<DropdownMenuTrigger asChild>
<Button
@@ -292,12 +297,19 @@ export default function DynamicFormItemComponent({
</div>
);
}
return <Input className="w-full max-w-md" {...field} />;
return (
<Input
className="w-full max-w-md"
{...field}
value={field.value ?? ''}
/>
);
case DynamicFormItemType.TEXT:
return (
<Textarea
{...field}
value={field.value ?? ''}
className="min-h-[120px] w-full max-w-full resize-y overflow-x-hidden break-all"
/>
);
@@ -306,13 +318,16 @@ export default function DynamicFormItemComponent({
return (
<Textarea
{...field}
value={field.value ?? ''}
className="min-h-[200px] font-mono text-sm"
placeholder='{"key": "value"}'
/>
);
case DynamicFormItemType.BOOLEAN:
return <Switch checked={field.value} onCheckedChange={field.onChange} />;
return (
<Switch checked={!!field.value} onCheckedChange={field.onChange} />
);
case DynamicFormItemType.STRING_ARRAY:
return (
@@ -1430,7 +1445,7 @@ export default function DynamicFormItemComponent({
{/* 内容输入 */}
<Textarea
className="min-h-20 w-full min-w-0 flex-1 resize-y overflow-x-hidden break-all sm:w-[300px]"
value={item.content}
value={item.content ?? ''}
onChange={(e) => {
const newValue = [...(field.value ?? promptItems)];
newValue[index] = {

View File

@@ -310,6 +310,7 @@ function SingleSelectField({
{options.map((opt) => (
<div key={opt.id}>
<button
type="button"
onClick={() => onChange(opt.id)}
className={`w-full text-left text-sm px-3 py-2 rounded-lg border transition-colors ${
value === opt.id
@@ -361,8 +362,16 @@ function MultiSelectField({
const selected = value.includes(opt.id);
return (
<div key={opt.id}>
<button
<div
role="button"
tabIndex={0}
onClick={() => toggle(opt.id)}
onKeyDown={(e) => {
if (e.key === 'Enter' || e.key === ' ') {
e.preventDefault();
toggle(opt.id);
}
}}
className={`w-full text-left text-sm px-3 py-2 rounded-lg border transition-colors flex items-center gap-2 ${
selected
? 'border-primary bg-primary/5 text-primary'
@@ -371,7 +380,7 @@ function MultiSelectField({
>
<Checkbox checked={selected} className="pointer-events-none" />
{getI18nText(opt.label)}
</button>
</div>
{opt.has_input && selected && (
<input
type="text"

View File

@@ -464,14 +464,11 @@ const MCPForm = forwardRef<MCPFormHandle, MCPFormProps>(function MCPForm(
onRuntimeInfoChange?.(runtimeInfo);
}, [onRuntimeInfoChange, runtimeInfo]);
useImperativeHandle(
ref,
() => ({
testMcp: () => testMcp(),
isTesting: mcpTesting,
}),
[mcpTesting],
);
// Expose test action and testing state to parent
useImperativeHandle(ref, () => ({
testMcp: () => testMcp(),
isTesting: mcpTesting,
}));
useEffect(() => {
isInitializing.current = true;

View File

@@ -169,6 +169,8 @@ export default function PipelineFormComponent({
resolver: zodResolver(formSchema),
defaultValues: {
basic: {
name: '',
description: '',
emoji: '⚙️',
},
ai: {},
@@ -221,8 +223,8 @@ export default function PipelineFormComponent({
const loadedValues = {
basic: {
name: resp.pipeline.name,
description: resp.pipeline.description,
name: resp.pipeline.name ?? '',
description: resp.pipeline.description ?? '',
emoji: resp.pipeline.emoji || '⚙️',
},
ai: resp.pipeline.config.ai,
@@ -601,7 +603,7 @@ export default function PipelineFormComponent({
<span className="text-destructive">*</span>
</FormLabel>
<FormControl>
<Input {...field} />
<Input {...field} value={field.value ?? ''} />
</FormControl>
<FormMessage />
</FormItem>
@@ -632,7 +634,7 @@ export default function PipelineFormComponent({
<FormItem>
<FormLabel>{t('common.description')}</FormLabel>
<FormControl>
<Input {...field} />
<Input {...field} value={field.value ?? ''} />
</FormControl>
<FormMessage />
</FormItem>