ppt-tool/frontend/app/ConfigurationInitializer.tsx
Vadym Samoilenko 8670822a59 Fix bare fetch() calls missing /ppt-tool basePath prefix
ConfigurationInitializer: /api/can-change-keys and /api/user-config
providerUtils: /api/v1/ppt/ollama/models/available

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-20 17:52:47 +00:00

145 lines
4.9 KiB
TypeScript

'use client';
import { useEffect, useState } from 'react';
import { apiFetch } from '../lib/apiFetch';
import { setCanChangeKeys, setLLMConfig } from '@/store/slices/userConfig';
import { hasValidLLMConfig } from '@/utils/storeHelpers';
import { usePathname, useRouter } from 'next/navigation';
import { useDispatch } from 'react-redux';
import { checkIfSelectedOllamaModelIsPulled } from '@/utils/providerUtils';
import { LLMConfig } from '@/types/llm_config';
export function ConfigurationInitializer({ children }: { children: React.ReactNode }) {
const dispatch = useDispatch();
const [isLoading, setIsLoading] = useState(true);
const router = useRouter();
const route = usePathname();
// Fetch user config state
useEffect(() => {
fetchUserConfigState();
}, []);
const setLoadingToFalseAfterNavigatingTo = (pathname: string) => {
const interval = setInterval(() => {
if (window.location.pathname === pathname) {
clearInterval(interval);
setIsLoading(false);
}
}, 500);
}
const fetchUserConfigState = async () => {
setIsLoading(true);
const response = await apiFetch('/api/can-change-keys');
const canChangeKeys = (await response.json()).canChange;
dispatch(setCanChangeKeys(canChangeKeys));
if (canChangeKeys) {
const response = await apiFetch('/api/user-config');
const llmConfig = await response.json();
if (!llmConfig.LLM) {
llmConfig.LLM = 'openai';
}
dispatch(setLLMConfig(llmConfig));
const isValid = hasValidLLMConfig(llmConfig);
if (isValid) {
// Check if the selected Ollama model is pulled
if (llmConfig.LLM === 'ollama') {
const isPulled = await checkIfSelectedOllamaModelIsPulled(llmConfig.OLLAMA_MODEL);
if (!isPulled) {
router.push('/');
setLoadingToFalseAfterNavigatingTo('/');
return;
}
}
if (llmConfig.LLM === 'custom') {
const isAvailable = await checkIfSelectedCustomModelIsAvailable(llmConfig);
if (!isAvailable) {
router.push('/');
setLoadingToFalseAfterNavigatingTo('/');
return;
}
}
if (route === '/') {
router.push('/dashboard');
setLoadingToFalseAfterNavigatingTo('/dashboard');
} else {
setIsLoading(false);
}
} else if (route !== '/') {
router.push('/');
setLoadingToFalseAfterNavigatingTo('/');
} else {
setIsLoading(false);
}
} else {
if (route === '/') {
router.push('/dashboard');
setLoadingToFalseAfterNavigatingTo('/dashboard');
} else {
setIsLoading(false);
}
}
}
const checkIfSelectedCustomModelIsAvailable = async (llmConfig: LLMConfig) => {
try {
const response = await apiFetch('/api/v1/ppt/openai/models/available', {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({
url: llmConfig.CUSTOM_LLM_URL,
api_key: llmConfig.CUSTOM_LLM_API_KEY,
}),
});
const data = await response.json();
return data.includes(llmConfig.CUSTOM_MODEL);
} catch (error) {
console.error('Error fetching custom models:', error);
return false;
}
}
if (isLoading) {
return (
<div className="min-h-screen bg-gradient-to-br from-[#E9E8F8] via-[#F5F4FF] to-[#E0DFF7] flex items-center justify-center p-4">
<div className="max-w-md w-full">
<div className="bg-white/80 backdrop-blur-sm rounded-2xl shadow-xl border border-white/20 p-8 text-center">
{/* Logo/Branding */}
<div className="mb-6">
<h2 className="text-xl font-bold text-gray-900 font-inter mb-4">Oliver DeckForge</h2>
<div className="w-16 h-1 bg-gradient-to-r from-blue-500 to-purple-600 mx-auto rounded-full"></div>
</div>
{/* Loading Text */}
<div className="space-y-2">
<h3 className="text-lg font-semibold text-gray-800 font-inter">
Initializing Application
</h3>
<p className="text-sm text-gray-600 font-inter">
Loading configuration and checking model availability...
</p>
</div>
{/* Progress Indicator */}
<div className="mt-6">
<div className="flex space-x-1 justify-center">
<div className="w-2 h-2 bg-blue-500 rounded-full animate-pulse"></div>
<div className="w-2 h-2 bg-purple-500 rounded-full animate-pulse" style={{ animationDelay: '0.2s' }}></div>
<div className="w-2 h-2 bg-blue-500 rounded-full animate-pulse" style={{ animationDelay: '0.4s' }}></div>
</div>
</div>
</div>
</div>
</div>
);
}
return children;
}