Files
DataClaw/frontend/src/components/DataSourceForm.tsx
T

322 lines
11 KiB
TypeScript
Raw Normal View History

2026-03-15 20:48:40 +08:00
import { useState, useRef } from "react";
2026-03-15 19:36:02 +08:00
import { Button } from "@/components/ui/button";
import { Input } from "@/components/ui/input";
2026-03-15 20:48:40 +08:00
import { Loader2, Check, AlertTriangle, Upload } from "lucide-react";
import { api } from "@/lib/api";
2026-03-15 19:36:02 +08:00
export interface DataSourceConfig {
id?: number;
name: string;
type: string;
config: Record<string, any>;
}
interface DataSourceFormProps {
initialData?: DataSourceConfig | null;
onSubmit: (data: Omit<DataSourceConfig, "id">) => Promise<void>;
onTest: (type: string, config: Record<string, any>) => Promise<boolean>;
onCancel: () => void;
}
export function DataSourceForm({ initialData, onSubmit, onTest, onCancel }: DataSourceFormProps) {
const [name, setName] = useState(initialData?.name || "");
const [type, setType] = useState(initialData?.type || "postgres");
const [config, setConfig] = useState<Record<string, any>>(initialData?.config || {});
const [isTesting, setIsTesting] = useState(false);
const [testResult, setTestResult] = useState<{ success: boolean; message: string } | null>(null);
const [isSaving, setIsSaving] = useState(false);
2026-03-15 20:48:40 +08:00
const [isUploading, setIsUploading] = useState(false);
const fileInputRef = useRef<HTMLInputElement>(null);
2026-03-15 19:36:02 +08:00
const handleConfigChange = (key: string, value: any) => {
setConfig(prev => ({ ...prev, [key]: value }));
};
2026-03-15 20:48:40 +08:00
const handleFileSelect = () => {
fileInputRef.current?.click();
};
const handleFileUpload = async (e: React.ChangeEvent<HTMLInputElement>) => {
const file = e.target.files?.[0];
if (!file) return;
setIsUploading(true);
const formData = new FormData();
formData.append("file", file);
try {
// @ts-ignore
const res = await api.post("/api/v1/upload/file", formData);
if (res && (res as any).url) {
handleConfigChange("file_path", (res as any).url);
}
} catch (error) {
console.error("Upload failed", error);
alert("上传失败");
} finally {
setIsUploading(false);
// Clear input value so same file can be selected again
if (fileInputRef.current) {
fileInputRef.current.value = "";
}
}
};
2026-03-15 19:36:02 +08:00
const handleTest = async () => {
setIsTesting(true);
setTestResult(null);
try {
const success = await onTest(type, config);
setTestResult({
success,
message: success ? "连接成功" : "连接失败",
});
} catch (e: any) {
setTestResult({
success: false,
message: e.message || "连接失败",
});
} finally {
setIsTesting(false);
}
};
const handleSubmit = async (e: React.FormEvent) => {
e.preventDefault();
setIsSaving(true);
try {
await onSubmit({ name, type, config });
} finally {
setIsSaving(false);
}
};
const renderConfigFields = () => {
switch (type) {
case "postgres":
case "supabase":
return (
<div className="space-y-4">
<div className="grid grid-cols-2 gap-4">
<div className="space-y-2">
<label className="text-sm font-medium">Host</label>
<Input
value={config.host || ""}
onChange={e => handleConfigChange("host", e.target.value)}
placeholder="localhost"
/>
</div>
<div className="space-y-2">
<label className="text-sm font-medium">Port</label>
<Input
type="number"
value={config.port || 5432}
onChange={e => handleConfigChange("port", parseInt(e.target.value))}
placeholder="5432"
/>
</div>
</div>
<div className="space-y-2">
<label className="text-sm font-medium">Database</label>
<Input
value={config.database || ""}
onChange={e => handleConfigChange("database", e.target.value)}
placeholder="postgres"
/>
</div>
<div className="space-y-2">
<label className="text-sm font-medium">Username</label>
<Input
value={config.user || ""}
onChange={e => handleConfigChange("user", e.target.value)}
placeholder="postgres"
/>
</div>
<div className="space-y-2">
<label className="text-sm font-medium">Password</label>
<Input
type="password"
value={config.password || ""}
onChange={e => handleConfigChange("password", e.target.value)}
placeholder="••••••"
/>
</div>
<div className="text-xs text-zinc-500 pt-2">
使 ():
</div>
<div className="space-y-2">
<label className="text-sm font-medium">Connection String</label>
<Input
value={config.connection_string || ""}
onChange={e => handleConfigChange("connection_string", e.target.value)}
placeholder="postgresql://user:pass@host:5432/db"
/>
</div>
</div>
);
case "clickhouse":
return (
<div className="space-y-4">
<div className="grid grid-cols-2 gap-4">
<div className="space-y-2">
<label className="text-sm font-medium">Host</label>
<Input
value={config.host || ""}
onChange={e => handleConfigChange("host", e.target.value)}
placeholder="localhost"
/>
</div>
<div className="space-y-2">
<label className="text-sm font-medium">Port</label>
<Input
type="number"
value={config.port || 9000}
onChange={e => handleConfigChange("port", parseInt(e.target.value))}
placeholder="9000"
/>
</div>
</div>
<div className="space-y-2">
<label className="text-sm font-medium">Database</label>
<Input
value={config.database || ""}
onChange={e => handleConfigChange("database", e.target.value)}
placeholder="default"
/>
</div>
<div className="space-y-2">
<label className="text-sm font-medium">Username</label>
<Input
value={config.user || ""}
onChange={e => handleConfigChange("user", e.target.value)}
placeholder="default"
/>
</div>
<div className="space-y-2">
<label className="text-sm font-medium">Password</label>
<Input
type="password"
value={config.password || ""}
onChange={e => handleConfigChange("password", e.target.value)}
placeholder="••••••"
/>
</div>
</div>
);
case "sqlite":
return (
<div className="space-y-4">
<div className="space-y-2">
<label className="text-sm font-medium">File Path (Server Side)</label>
2026-03-15 20:48:40 +08:00
<div className="flex gap-2">
<Input
value={config.file_path || ""}
onChange={e => handleConfigChange("file_path", e.target.value)}
placeholder="/path/to/database.db"
/>
<Button type="button" variant="outline" onClick={handleFileSelect} disabled={isUploading}>
{isUploading ? <Loader2 className="h-4 w-4 animate-spin" /> : <Upload className="h-4 w-4" />}
</Button>
<input
key="sqlite-input"
type="file"
ref={fileInputRef}
className="hidden"
accept=".db,.sqlite,.sqlite3"
onChange={handleFileUpload}
/>
</div>
2026-03-15 19:36:02 +08:00
</div>
</div>
);
case "parquet":
return (
<div className="space-y-4">
<div className="space-y-2">
<label className="text-sm font-medium">File Path (Server Side)</label>
2026-03-15 20:48:40 +08:00
<div className="flex gap-2">
<Input
value={config.file_path || ""}
onChange={e => handleConfigChange("file_path", e.target.value)}
placeholder="/path/to/data.parquet"
/>
<Button type="button" variant="outline" onClick={handleFileSelect} disabled={isUploading}>
{isUploading ? <Loader2 className="h-4 w-4 animate-spin" /> : <Upload className="h-4 w-4" />}
</Button>
<input
key="parquet-input"
type="file"
ref={fileInputRef}
className="hidden"
accept=".parquet"
onChange={handleFileUpload}
/>
</div>
2026-03-15 19:36:02 +08:00
</div>
</div>
);
default:
return null;
}
};
return (
<form onSubmit={handleSubmit} className="space-y-6">
<div className="space-y-2">
<label className="text-sm font-medium"></label>
<Input
value={name}
onChange={e => setName(e.target.value)}
placeholder="我的数据源"
required
/>
</div>
<div className="space-y-2">
<label className="text-sm font-medium"></label>
<select
className="w-full h-10 px-3 rounded-md border border-zinc-200 bg-white text-sm focus:outline-none focus:ring-2 focus:ring-zinc-950 focus:border-transparent"
value={type}
onChange={e => setType(e.target.value)}
>
<option value="postgres">PostgreSQL</option>
<option value="clickhouse">ClickHouse</option>
<option value="sqlite">SQLite</option>
<option value="supabase">Supabase</option>
<option value="parquet">Parquet</option>
</select>
</div>
<div className="p-4 border border-zinc-200 rounded-lg bg-zinc-50/50">
{renderConfigFields()}
</div>
{testResult && (
<div className={`p-3 rounded-md flex items-center gap-2 text-sm ${testResult.success ? 'bg-green-50 text-green-700' : 'bg-red-50 text-red-700'}`}>
{testResult.success ? <Check className="h-4 w-4" /> : <AlertTriangle className="h-4 w-4" />}
{testResult.message}
</div>
)}
<div className="flex justify-end gap-3 pt-4">
<Button type="button" variant="outline" onClick={onCancel}>
</Button>
<Button
type="button"
variant="secondary"
onClick={handleTest}
disabled={isTesting}
>
{isTesting && <Loader2 className="mr-2 h-4 w-4 animate-spin" />}
</Button>
<Button type="submit" disabled={isSaving}>
{isSaving && <Loader2 className="mr-2 h-4 w-4 animate-spin" />}
</Button>
</div>
</form>
);
}