mirror of
https://github.com/ostris/ai-toolkit.git
synced 2026-01-26 16:39:47 +00:00
105 lines
3.7 KiB
TypeScript
105 lines
3.7 KiB
TypeScript
import { Job } from '@prisma/client';
|
|
import useGPUInfo from '@/hooks/useGPUInfo';
|
|
import GPUWidget from '@/components/GPUWidget';
|
|
import FilesWidget from '@/components/FilesWidget';
|
|
import { getJobConfig, getTotalSteps } from '@/utils/jobs';
|
|
import { Cpu, HardDrive, Info } from 'lucide-react';
|
|
import { useMemo } from 'react';
|
|
|
|
interface JobOverviewProps {
|
|
job: Job;
|
|
}
|
|
|
|
export default function JobOverview({ job }: JobOverviewProps) {
|
|
const gpuIds = useMemo(() => job.gpu_ids.split(',').map(id => parseInt(id)), [job.gpu_ids]);
|
|
|
|
const { gpuList, isGPUInfoLoaded } = useGPUInfo(gpuIds, 5000);
|
|
const totalSteps = getTotalSteps(job);
|
|
const progress = (job.step / totalSteps) * 100;
|
|
const isStopping = job.stop && job.status === 'running';
|
|
|
|
const getStatusColor = (status: string) => {
|
|
switch (status.toLowerCase()) {
|
|
case 'running':
|
|
return 'bg-emerald-500/10 text-emerald-500';
|
|
case 'stopping':
|
|
return 'bg-amber-500/10 text-amber-500';
|
|
case 'stopped':
|
|
return 'bg-gray-500/10 text-gray-400';
|
|
case 'completed':
|
|
return 'bg-blue-500/10 text-blue-500';
|
|
case 'error':
|
|
return 'bg-rose-500/10 text-rose-500';
|
|
default:
|
|
return 'bg-gray-500/10 text-gray-400';
|
|
}
|
|
};
|
|
|
|
let status = job.status;
|
|
if (isStopping) {
|
|
status = 'stopping';
|
|
}
|
|
|
|
return (
|
|
<div className="grid grid-cols-1 gap-6 md:grid-cols-3">
|
|
{/* Job Information Panel */}
|
|
<div className="col-span-2 bg-gray-900 rounded-xl shadow-lg overflow-hidden border border-gray-800">
|
|
<div className="bg-gray-800 px-4 py-3 flex items-center justify-between">
|
|
<h2 className="font-semibold text-gray-100">Job Details</h2>
|
|
<span className={`px-3 py-1 rounded-full text-sm ${getStatusColor(job.status)}`}>{job.status}</span>
|
|
</div>
|
|
|
|
<div className="p-4 space-y-6">
|
|
{/* Progress Bar */}
|
|
<div className="space-y-2">
|
|
<div className="flex items-center justify-between text-sm">
|
|
<span className="text-gray-400">Progress</span>
|
|
<span className="text-gray-200">
|
|
Step {job.step} of {totalSteps}
|
|
</span>
|
|
</div>
|
|
<div className="w-full bg-gray-800 rounded-full h-2">
|
|
<div className="h-2 rounded-full bg-blue-500 transition-all" style={{ width: `${progress}%` }} />
|
|
</div>
|
|
</div>
|
|
|
|
{/* Job Info Grid */}
|
|
<div className="grid gap-4">
|
|
<div className="flex items-center space-x-4">
|
|
<HardDrive className="w-5 h-5 text-blue-400" />
|
|
<div>
|
|
<p className="text-xs text-gray-400">Job Name</p>
|
|
<p className="text-sm font-medium text-gray-200">{job.name}</p>
|
|
</div>
|
|
</div>
|
|
|
|
<div className="flex items-center space-x-4">
|
|
<Cpu className="w-5 h-5 text-purple-400" />
|
|
<div>
|
|
<p className="text-xs text-gray-400">Assigned GPUs</p>
|
|
<p className="text-sm font-medium text-gray-200">GPUs: {job.gpu_ids}</p>
|
|
</div>
|
|
</div>
|
|
|
|
<div className="flex items-center space-x-4">
|
|
<Info className="w-5 h-5 text-amber-400" />
|
|
<div>
|
|
<p className="text-xs text-gray-400">Additional Information</p>
|
|
<p className="text-sm font-medium text-gray-200">{job.info}</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
{/* GPU Widget Panel */}
|
|
<div className="col-span-1">
|
|
<div>{isGPUInfoLoaded && gpuList.length > 0 && <GPUWidget gpu={gpuList[0]} />}</div>
|
|
<div className="mt-4">
|
|
<FilesWidget jobID={job.id} />
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|