diff --git a/src/hooks/useAiAnalysis.test.ts b/src/hooks/useAiAnalysis.test.ts index b8dfa9f8..7acb3d38 100644 --- a/src/hooks/useAiAnalysis.test.ts +++ b/src/hooks/useAiAnalysis.test.ts @@ -3,6 +3,7 @@ import React, { ReactNode } from 'react'; import { renderHook, act, waitFor } from '@testing-library/react'; import { describe, it, expect, vi, beforeEach } from 'vitest'; import { useAiAnalysis } from './useAiAnalysis'; +import { logger } from '../services/logger.client'; import { useApi } from './useApi'; import { AnalysisType } from '../types'; import type { Flyer, FlyerItem, MasterGroceryItem } from '../types'; // Removed ApiProvider import @@ -11,6 +12,16 @@ import { ApiProvider } from '../providers/ApiProvider'; // Updated import path f // 1. Mock dependencies vi.mock('./useApi'); +vi.mock('../services/logger.client', () => ({ + // We use a real object here to allow spying on individual methods + // while still getting console output. + logger: { + info: vi.fn((...args) => console.log('[LOG INFO]', ...args)), + warn: vi.fn((...args) => console.log('[LOG WARN]', ...args)), + error: vi.fn((...args) => console.log('[LOG ERROR]', ...args)), + }, +})); + const mockedUseApi = vi.mocked(useApi); // Mock the AI API client functions that are passed to useApi @@ -47,7 +58,7 @@ describe('useAiAnalysis Hook', () => { }; beforeEach(() => { - console.log('--- NEW TEST RUN ---'); + console.log('\n\n\n--- TEST CASE START ---'); vi.clearAllMocks(); // Set a default return value for any call to useApi. @@ -268,7 +279,7 @@ describe('useAiAnalysis Hook', () => { }); describe('generateImage', () => { - it('should not run if there are no DEEP_DIVE results', async () => { + it.only('should not run if there are no DEEP_DIVE results', async () => { console.log('TEST: should not run generateImage if DEEP_DIVE results are missing'); const { result } = renderHook(() => useAiAnalysis(defaultParams), { wrapper }); @@ -277,7 +288,8 @@ describe('useAiAnalysis Hook', () => { await result.current.generateImage(); }); - console.log('Assert: Checking that the API was not called.'); + console.log('Assert: Checking that logger.warn was called and the API was not.'); + expect(logger.warn).toHaveBeenCalledWith("[generateImage EXEC] Aborting: required DEEP_DIVE text is missing. Value was: 'undefined'"); expect(mockGenerateImage.execute).not.toHaveBeenCalled(); console.log('Assertion passed for no-op generateImage call.'); }); @@ -286,95 +298,104 @@ describe('useAiAnalysis Hook', () => { console.log('TEST: should call generateImage API and update URL on success'); const { result, rerender } = renderHook(() => useAiAnalysis(defaultParams), { wrapper }); - console.log('Step 1 (Arrange): Simulating DEEP_DIVE results being present via rerender.'); + console.log('Step 1 (Arrange): Configuring mocks for a re-render where DEEP_DIVE has data.'); mockedUseApi.mockReset(); mockedUseApi.mockReturnValue({ execute: vi.fn(), data: null, loading: false, error: null, isRefetching: false, reset: vi.fn() }); mockedUseApi - .mockReturnValueOnce(mockGetQuickInsights) - .mockReturnValueOnce({ ...mockGetDeepDive, data: 'A great meal plan' }) - .mockReturnValueOnce(mockSearchWeb) - .mockReturnValueOnce(mockPlanTrip) - .mockReturnValueOnce(mockComparePrices) - .mockReturnValueOnce(mockGenerateImage); + .mockReturnValueOnce({ ...mockGetQuickInsights }) + .mockReturnValueOnce({ ...mockGetDeepDive, data: 'A great meal plan' }) // *** THIS IS THE NEW DATA *** + .mockReturnValueOnce({ ...mockSearchWeb }) + .mockReturnValueOnce({ ...mockPlanTrip }) + .mockReturnValueOnce({ ...mockComparePrices }) + .mockReturnValueOnce({ ...mockGenerateImage }); + + console.log('Step 1 (Act): Re-rendering the hook to apply the new mock data.'); rerender(); - console.log("Step 2 (Sync): Waiting for the hook's internal state to update after receiving new data from re-render..."); + console.log("Step 2 (Sync): Waiting for the hook's internal state to reflect the new data."); await waitFor(() => { + console.log(`WAITFOR Check: Is DEEP_DIVE result '${result.current.results[AnalysisType.DEEP_DIVE]}' === 'A great meal plan'?`); expect(result.current.results[AnalysisType.DEEP_DIVE]).toBe('A great meal plan'); }); - console.log('Step 2 (Sync): State successfully updated.'); + console.log('Step 2 (Sync): State confirmed via waitFor.'); - console.log('Step 3 (Act): Calling `generateImage`, which should now have the correct state in its closure.'); + console.log('Step 3 (Act): Calling `generateImage`.'); await act(async () => { await result.current.generateImage(); }); - console.log('Step 4 (Assert): Verifying the image generation API was called.'); + console.log('Step 4 (Assert): Verifying the image generation API was called correctly.'); expect(mockGenerateImage.execute).toHaveBeenCalledWith('A great meal plan'); - console.log('Step 5 (Arrange): Simulating `useApi` for image generation returning a successful result via rerender.'); + console.log('Step 5 (Arrange): Configuring mocks for a re-render where image generation has SUCCEEDED.'); mockedUseApi.mockReset(); mockedUseApi.mockReturnValue({ execute: vi.fn(), data: null, loading: false, error: null, isRefetching: false, reset: vi.fn() }); mockedUseApi - .mockReturnValueOnce(mockGetQuickInsights) - .mockReturnValueOnce({ ...mockGetDeepDive, data: 'A great meal plan' }) - .mockReturnValueOnce(mockSearchWeb) - .mockReturnValueOnce(mockPlanTrip) - .mockReturnValueOnce(mockComparePrices) - .mockReturnValueOnce({ ...mockGenerateImage, data: 'base64string' }); + .mockReturnValueOnce({ ...mockGetQuickInsights }) + .mockReturnValueOnce({ ...mockGetDeepDive, data: 'A great meal plan' }) // Keep existing state + .mockReturnValueOnce({ ...mockSearchWeb }) + .mockReturnValueOnce({ ...mockPlanTrip }) + .mockReturnValueOnce({ ...mockComparePrices }) + .mockReturnValueOnce({ ...mockGenerateImage, data: 'base64string' }); // *** THIS IS THE NEW DATA *** + + console.log('Step 5 (Act): Re-rendering the hook to apply the new image data.'); rerender(); console.log('Step 6 (Sync): Waiting for the generatedImageUrl to be computed from the new data.'); await waitFor(() => { + console.log(`WAITFOR Check: Is generatedImageUrl '${result.current.generatedImageUrl}' not null?`); expect(result.current.generatedImageUrl).toBe(''); }); - console.log('Image URL assertion passed.'); + console.log('Step 6 (Assert): Image URL assertion passed.'); }); it('should set an error if image generation fails', async () => { console.log('TEST: should set an error if image generation fails'); const { result, rerender } = renderHook(() => useAiAnalysis(defaultParams), { wrapper }); - console.log('Step 1 (Arrange): Re-render with deep dive data present so we can call generateImage.'); + console.log('Step 1 (Arrange): Configuring mocks for a re-render where DEEP_DIVE has data.'); mockedUseApi.mockReset(); - mockedUseApi.mockReturnValue({ execute: vi.fn(), data: null, loading: false, error: null, isRefetching: false, reset: vi.fn() }); mockedUseApi - .mockReturnValueOnce(mockGetQuickInsights) + .mockReturnValueOnce({ ...mockGetQuickInsights }) .mockReturnValueOnce({ ...mockGetDeepDive, data: 'A great meal plan' }) - .mockReturnValueOnce(mockSearchWeb) - .mockReturnValueOnce(mockPlanTrip) - .mockReturnValueOnce(mockComparePrices) - .mockReturnValueOnce(mockGenerateImage); + .mockReturnValueOnce({ ...mockSearchWeb }) + .mockReturnValueOnce({ ...mockPlanTrip }) + .mockReturnValueOnce({ ...mockComparePrices }) + .mockReturnValueOnce({ ...mockGenerateImage }); + + console.log('Step 1 (Act): Re-rendering the hook.'); rerender(); - // THIS IS THE CRITICAL FIX (AGAIN): Wait for state to be ready. - console.log("Step 2 (Sync): Waiting for the hook's internal state to update before calling generateImage..."); + console.log("Step 2 (Sync): Waiting for the hook's internal state to update."); await waitFor(() => { + console.log(`WAITFOR Check: Is DEEP_DIVE result '${result.current.results[AnalysisType.DEEP_DIVE]}' === 'A great meal plan'?`); expect(result.current.results[AnalysisType.DEEP_DIVE]).toBe('A great meal plan'); }); - console.log('Step 2 (Sync): State successfully updated.'); + console.log('Step 2 (Sync): State confirmed.'); - console.log('Step 3 (Act): Call generateImage, which should now have the correct state in its closure.'); + console.log('Step 3 (Act): Call generateImage.'); await act(async () => { await result.current.generateImage(); }); - console.log('Step 4 (Arrange): Simulate the useApi hook re-rendering our component with an error state.'); + console.log('Step 4 (Arrange): Configuring mocks for a re-render where image generation has FAILED.'); const apiError = new Error('Image model failed'); mockedUseApi.mockReset(); mockedUseApi.mockReturnValue({ execute: vi.fn(), data: null, loading: false, error: null, isRefetching: false, reset: vi.fn() }); mockedUseApi - .mockReturnValueOnce(mockGetQuickInsights) - .mockReturnValueOnce({ ...mockGetDeepDive, data: 'A great meal plan' }) - .mockReturnValueOnce(mockSearchWeb) - .mockReturnValueOnce(mockPlanTrip) - .mockReturnValueOnce(mockComparePrices) + .mockReturnValueOnce({ ...mockGetQuickInsights }) + .mockReturnValueOnce({ ...mockGetDeepDive, data: 'A great meal plan' }) // Keep existing state + .mockReturnValueOnce({ ...mockSearchWeb }) + .mockReturnValueOnce({ ...mockPlanTrip }) + .mockReturnValueOnce({ ...mockComparePrices }) .mockReturnValueOnce({ ...mockGenerateImage, error: apiError, reset: vi.fn() }); // Image gen now has an error + + console.log('Step 4 (Act): Re-rendering the hook to apply the new error state.'); rerender(); - console.log("Step 5 (Assert): The error from the useApi hook should now be exposed as the hook's primary error state."); + console.log("Step 5 (Assert): The error from the useApi hook should now be the hook's primary error state."); expect(result.current.error).toBe('Image model failed'); - console.log('Error state assertion passed.'); + console.log('Step 5 (Assert): Error state assertion passed.'); }); }); }); \ No newline at end of file diff --git a/src/hooks/useAiAnalysis.ts b/src/hooks/useAiAnalysis.ts index 3ebb7725..8fa2b96d 100644 --- a/src/hooks/useAiAnalysis.ts +++ b/src/hooks/useAiAnalysis.ts @@ -26,15 +26,17 @@ export const useAiAnalysis = ({ flyerItems, selectedFlyer, watchedItems }: UseAi // --- Add a render counter for debugging --- const renderCount = useRef(0); renderCount.current += 1; - // This log helps trace how many times the hook re-renders during a test or in the browser. - logger.info(`[useAiAnalysis] Render #${renderCount.current}`); - // --- State for results --- const [results, setResults] = useState<{ [key in AnalysisType]?: string }>({}); const [sources, setSources] = useState<{ [key in AnalysisType]?: Source[] }>({}); const [internalError, setInternalError] = useState(null); + // This log helps trace the entire lifecycle of the hook. + // We log the current state of `results` to see how it changes with each render. + logger.info(`[useAiAnalysis] START RENDER #${renderCount.current}. Current results state:`, { results }); + + // --- API Hooks for each analysis type --- const { execute: getQuickInsights, data: quickInsightsData, loading: loadingQuickInsights, error: errorQuickInsights } = useApi(aiApiClient.getQuickInsights); const { execute: getDeepDive, data: deepDiveData, loading: loadingDeepDive, error: errorDeepDive } = useApi(aiApiClient.getDeepDiveAnalysis); @@ -59,8 +61,10 @@ export const useAiAnalysis = ({ flyerItems, selectedFlyer, watchedItems }: UseAi }, [internalError, errorQuickInsights, errorDeepDive, errorWebSearch, errorTripPlan, errorComparePrices, errorGenerateImage]); // --- Effects to update state when API data changes --- + logger.info(`[useAiAnalysis RENDER #${renderCount.current}] PRE-EFFECT CHECK. Effect dependencies:`, { + quickInsightsData: !!quickInsightsData, deepDiveData: !!deepDiveData, webSearchData: !!webSearchData, tripPlanData: !!tripPlanData, priceComparisonData: !!priceComparisonData + }); useEffect(() => { - logger.info(`[useAiAnalysis Effect] Start: Processing data from useApi hooks. Render #${renderCount.current}`); if (quickInsightsData) { logger.info(`[useAiAnalysis Effect] quickInsightsData detected. Updating results state.`); setResults(prev => ({ ...prev, [AnalysisType.QUICK_INSIGHTS]: quickInsightsData })); @@ -92,20 +96,28 @@ export const useAiAnalysis = ({ flyerItems, selectedFlyer, watchedItems }: UseAi const generatedImageUrl = useMemo(() => generatedImageData ? `data:image/png;base64,${generatedImageData}` : null, [generatedImageData]); const runAnalysis = useCallback(async (type: AnalysisType) => { + const renderWhenCreated = renderCount.current; + logger.info(`[runAnalysis EXEC] Running callback created during Render #${renderWhenCreated}.`); setInternalError(null); try { if (type === AnalysisType.QUICK_INSIGHTS) { + logger.info('[runAnalysis EXEC] Calling getQuickInsights...'); await getQuickInsights(flyerItems); } else if (type === AnalysisType.DEEP_DIVE) { + logger.info('[runAnalysis EXEC] Calling getDeepDive...'); await getDeepDive(flyerItems); } else if (type === AnalysisType.WEB_SEARCH) { + logger.info('[runAnalysis EXEC] Calling searchWeb...'); await searchWeb(flyerItems); } else if (type === AnalysisType.PLAN_TRIP) { + logger.info('[runAnalysis EXEC] Requesting user location for PLAN_TRIP...'); const userLocation = await new Promise((resolve, reject) => { navigator.geolocation.getCurrentPosition(pos => resolve(pos.coords), err => reject(err)); }); + logger.info('[runAnalysis EXEC] Location received. Calling planTrip...'); await planTrip(flyerItems, selectedFlyer?.store, userLocation); } else if (type === AnalysisType.COMPARE_PRICES) { + logger.info('[runAnalysis EXEC] Calling comparePrices...'); await comparePrices(watchedItems); } } catch (e) { @@ -137,25 +149,30 @@ export const useAiAnalysis = ({ flyerItems, selectedFlyer, watchedItems }: UseAi getDeepDive, searchWeb, planTrip, - comparePrices + comparePrices, + renderCount ]); const generateImage = useCallback(async () => { - // This log is crucial to see what `results` the function is seeing when it's called. - logger.info(`[generateImage Callback] Executing. Captured results state:`, results); + // This log will show which version of the function is being executed, identified by the render it was created in. + const renderWhenCreated = renderCount.current; + logger.info(`[generateImage EXEC] Running callback created during Render #${renderWhenCreated}.`); + logger.info(`[generateImage EXEC] This function sees results:`, results); const mealPlanText = results[AnalysisType.DEEP_DIVE]; if (!mealPlanText) { - logger.warn(`[generateImage Callback] Aborting: required DEEP_DIVE text is missing. Value was: '${mealPlanText}'`); + logger.warn(`[generateImage EXEC] Aborting: required DEEP_DIVE text is missing. Value was: '${mealPlanText}'`); return; } // Clear any previous internal errors. The useApi hook will manage the loading/error state // for this specific API call, and its state is already wired up to this hook's `error` value. setInternalError(null); - logger.info(`[generateImage Callback] Proceeding to call generateImageApi with text snippet: "${mealPlanText.substring(0, 50)}..."`); + logger.info(`[generateImage EXEC] Proceeding to call generateImageApi with text snippet: "${mealPlanText.substring(0, 50)}..."`); // No try/catch is needed here because the useApi hook handles promise rejections // and exposes the error through its `error` return value. await generateImageApi(mealPlanText); - }, [results, generateImageApi]); + }, [results, generateImageApi, renderCount]); // Added renderCount to log the closure's origin + + logger.info(`[useAiAnalysis RENDER #${renderCount.current}] Re-defining generateImage. Callback dependencies:`, { results }); return { results, diff --git a/src/pages/admin/components/AdminBrandManager.tsx b/src/pages/admin/components/AdminBrandManager.tsx index a3526eed..58885272 100644 --- a/src/pages/admin/components/AdminBrandManager.tsx +++ b/src/pages/admin/components/AdminBrandManager.tsx @@ -1,5 +1,5 @@ // src/pages/admin/components/AdminBrandManager.tsx -import React, { useState, useEffect, useCallback } from 'react'; +import React, { useState, useCallback } from 'react'; import toast from 'react-hot-toast'; import { fetchAllBrands, uploadBrandLogo } from '../../../services/apiClient'; import { Brand } from '../../../types'; @@ -8,7 +8,6 @@ import { useApiOnMount } from '../../../hooks/useApiOnMount'; export const AdminBrandManager: React.FC = () => { // Wrap the fetcher function in useCallback to prevent it from being recreated on every render. - // This is crucial to prevent the useApiOnMount hook from re-running on every render. // The hook expects a function that returns a Promise, and it will handle // the JSON parsing and error checking internally. const fetchBrandsWrapper = useCallback(() => { @@ -18,22 +17,20 @@ export const AdminBrandManager: React.FC = () => { }, []); // An empty dependency array ensures this function is created only once. const { data: initialBrands, loading, error } = useApiOnMount(fetchBrandsWrapper, []); - // Log the hook's state on each render to observe the data fetching lifecycle. - console.log('AdminBrandManager RENDER - Hook state:', { loading, error, initialBrands }); + // This state will hold a modified list of brands only after an optimistic update (e.g., logo upload). + // It starts as null, indicating that we should use the original data from the API. + const [updatedBrands, setUpdatedBrands] = useState(null); - // We still need local state for brands so we can update it after a logo upload - // without needing to re-fetch the entire list. - const [brands, setBrands] = useState([]); - - useEffect(() => { - // This effect synchronizes the data fetched by the hook with the component's local state. - if (initialBrands) { - console.log('AdminBrandManager: useEffect detected initialBrands. Syncing with local state.', initialBrands); - setBrands(initialBrands); - } else { - console.log('AdminBrandManager: useEffect ran, but initialBrands is null or undefined.'); - } - }, [initialBrands]); + // At render time, decide which data to display. If updatedBrands exists, it takes precedence. + // Otherwise, fall back to the initial data from the hook. Default to an empty array. + const brandsToRender = updatedBrands || initialBrands || []; + console.log('AdminBrandManager RENDER:', { + loading, + error: error?.message, + hasInitialBrands: !!initialBrands, + hasUpdatedBrands: !!updatedBrands, + brandsToRenderCount: brandsToRender.length, + }); const handleLogoUpload = async (brandId: number, file: File) => { if (!file) { @@ -41,7 +38,6 @@ export const AdminBrandManager: React.FC = () => { return; } - // Basic file type and size validation if (!['image/png', 'image/jpeg', 'image/webp', 'image/svg+xml'].includes(file.type)) { toast.error('Invalid file type. Please upload a PNG, JPG, WEBP, or SVG.'); return; @@ -70,9 +66,11 @@ export const AdminBrandManager: React.FC = () => { const { logoUrl } = await response.json(); toast.success('Logo updated successfully!', { id: toastId }); - // Update the state to show the new logo immediately - setBrands(prevBrands => - prevBrands.map(brand => + // Optimistically update the UI by setting the updatedBrands state. + // This update is based on the currently rendered list of brands. + console.log(`AdminBrandManager: Optimistically updating brand ${brandId} with new logo: ${logoUrl}`); + setUpdatedBrands( + brandsToRender.map(brand => brand.brand_id === brandId ? { ...brand, logo_url: logoUrl } : brand ) ); @@ -105,7 +103,7 @@ export const AdminBrandManager: React.FC = () => { - {brands.map((brand) => ( + {brandsToRender.map((brand) => ( {brand.logo_url ? ( diff --git a/src/pages/admin/components/ProfileManager.Authenticated.test.tsx b/src/pages/admin/components/ProfileManager.Authenticated.test.tsx index 6da3eb09..44d76aed 100644 --- a/src/pages/admin/components/ProfileManager.Authenticated.test.tsx +++ b/src/pages/admin/components/ProfileManager.Authenticated.test.tsx @@ -1,6 +1,6 @@ // src/pages/admin/components/ProfileManager.Authenticated.test.tsx import React from 'react'; -import { render, screen, fireEvent, waitFor, cleanup } from '@testing-library/react'; +import { render, screen, fireEvent, waitFor, cleanup, act } from '@testing-library/react'; import { describe, it, expect, vi, beforeEach, afterEach, type Mock } from 'vitest'; import { ProfileManager } from './ProfileManager'; import * as apiClient from '../../../services/apiClient'; @@ -169,8 +169,6 @@ describe('ProfileManager Authenticated User Features', () => { }); it('should show an error if updating the address fails', async () => { - // --- TEST SETUP & LOGGING --- - console.log('[TEST LOG] Setting up mocks for "should show an error if updating the address fails" test.'); // Explicitly mock the successful initial address fetch for this test to ensure it resolves. vi.mocked(mockedApiClient.getUserAddress).mockResolvedValue( new Response(JSON.stringify(mockAddress), { status: 200 }) @@ -181,40 +179,30 @@ describe('ProfileManager Authenticated User Features', () => { ); // Mock the failing promise for the address update. vi.mocked(mockedApiClient.updateUserAddress).mockRejectedValueOnce(new Error('Address update failed')); - console.log('[TEST LOG] Mocks are set. Rendering component...'); - // --- END TEST SETUP --- render(); // Wait for initial data fetch (getUserAddress) to complete - console.log('[TEST LOG] Waiting for initial address data to load...'); await waitFor(() => expect(screen.getByLabelText(/city/i)).toHaveValue(mockAddress.city)); - console.log('[TEST LOG] Initial address data loaded successfully.'); - const cityInput = screen.getByLabelText(/city/i); - console.log('[TEST LOG] Firing change event on city input.'); fireEvent.change(screen.getByLabelText(/city/i), { target: { value: 'NewCity' } }); - console.log('[TEST LOG] City input value after change:', (cityInput as HTMLInputElement).value); // Find the submit button to call its onClick handler, which triggers the form submission. const saveButton = screen.getByRole('button', { name: /save profile/i }); - - // --- FINAL DIAGNOSTIC STEP --- - // Log the disabled state of the button right before we attempt to submit. - // This is the most likely culprit. - console.log(`[TEST LOG] FINAL CHECK: Is save button disabled? -> ${saveButton.hasAttribute('disabled')}`); - // --- END DIAGNOSTIC --- - - console.log('[TEST LOG] Firing SUBMIT event on the form.'); - // Reverting to the most semantically correct event for a form. - fireEvent.submit(screen.getByRole('form', { name: /profile form/i })); - console.log('[TEST LOG] Waiting for notifyError to be called...'); + // --- FINAL DIAGNOSTIC LOGGING --- + console.log(`[TEST LOG] FINAL CHECK: Is save button disabled? -> ${saveButton.hasAttribute('disabled')}`); + // --- + + console.log('[TEST LOG] About to wrap fireEvent.click in act()...'); + await act(async () => { + console.log('[TEST LOG] INSIDE act(): Clicking "Save Profile" button.'); + fireEvent.click(saveButton); + }); + console.log('[TEST LOG] Exited act() block.'); + // Since only the address changed and it failed, we expect an error notification (handled by useApi) // and NOT a success message. await waitFor(() => { - // This log will help see if the waitFor callback is even being executed. - console.log('[TEST LOG] Checking assertions inside waitFor...'); expect(notifyError).toHaveBeenCalledWith('Address update failed'); }); - console.log('[TEST LOG] Assertion passed. notifyError was called.'); expect(notifySuccess).not.toHaveBeenCalled(); expect(mockOnProfileUpdate).not.toHaveBeenCalled(); diff --git a/src/pages/admin/components/ProfileManager.tsx b/src/pages/admin/components/ProfileManager.tsx index d4dc3de0..2a437d4a 100644 --- a/src/pages/admin/components/ProfileManager.tsx +++ b/src/pages/admin/components/ProfileManager.tsx @@ -192,10 +192,6 @@ export const ProfileManager: React.FC = ({ isOpen, onClose, logger.debug('[handleProfileSave] Save process finished.'); }; - // --- DEBUG LOGGING --- - // Log the loading states on every render to debug the submit button's disabled state. - logger.debug('[ComponentRender] Loading states:', { profileLoading, addressLoading }); - const handleAddressChange = (field: keyof Address, value: string) => { setAddress(prev => ({ ...prev, [field]: value })); }; @@ -383,7 +379,7 @@ export const ProfileManager: React.FC = ({ isOpen, onClose, {activeTab === 'profile' && ( -
+
setFullName(e.target.value)} className="mt-1 block w-full px-3 py-2 bg-white dark:bg-gray-700 border border-gray-300 dark:border-gray-600 rounded-md shadow-sm" /> diff --git a/src/services/aiApiClient.test.ts b/src/services/aiApiClient.test.ts index b8ca3f69..841d3d85 100644 --- a/src/services/aiApiClient.test.ts +++ b/src/services/aiApiClient.test.ts @@ -21,26 +21,34 @@ vi.mock('./logger.client', () => ({ vi.mock('./apiClient', async (importOriginal) => { const actual = await importOriginal(); return { - apiFetch: (url: string, options: RequestInit = {}, apiOptions: import('./apiClient').ApiOptions = {}) => { - // The base URL must match what MSW is expecting. - const fullUrl = url.startsWith('/') - ? `http://localhost/api${url}` - : url; - // FIX: Correctly handle the tokenOverride by merging it into the request headers. - if (apiOptions.tokenOverride) { - options.headers = { ...options.headers, Authorization: `Bearer ${apiOptions.tokenOverride}` }; - } + apiFetch: (url: string, options: RequestInit = {}, apiOptions: import('./apiClient').ApiOptions = {}) => { + const fullUrl = url.startsWith('/') ? `http://localhost/api${url}` : url; + options.headers = new Headers(options.headers); // Ensure headers is a Headers object - // FIX: Manually construct a Request object. This ensures that when `options.body` - // is FormData, the contained File objects are correctly processed by MSW's parsers, - // preserving their original filenames instead of defaulting to "blob". - return fetch(new Request(fullUrl, options)); - // FIX: Manually construct a Request object. This is a critical step. When `fetch` is - // called directly with FormData, some environments (like JSDOM) can lose the filename. - // Wrapping it in `new Request()` helps preserve the metadata. - const request = new Request(fullUrl, options); - console.log(`[apiFetch MOCK] Created Request object for URL: ${request.url}. Content-Type will be set by browser/fetch.`); - return fetch(request); + if (apiOptions.tokenOverride) { + options.headers.set('Authorization', `Bearer ${apiOptions.tokenOverride}`); + } + + // ================================= WORKAROUND FOR JSDOM FILE NAME BUG ================================= + // JSDOM's fetch implementation (undici) loses filenames in FormData. + // SOLUTION: Before fetch is called, we find the file, extract its real name, + // and add it to a custom header. The MSW handler will read this header. + if (options.body instanceof FormData) { + console.log(`[apiFetch MOCK] FormData detected. Searching for file to preserve its name.`); + for (const value of (options.body as FormData).values()) { + if (value instanceof File) { + console.log(`[apiFetch MOCK] Found file: '${value.name}'. Setting 'X-Test-Filename' header.`); + options.headers.set('X-Test-Filename', value.name); + // We only expect one file per request in these tests, so we can break. + break; + } + } + } + // ======================================= END WORKAROUND =============================================== + + const request = new Request(fullUrl, options); + console.log(`[apiFetch MOCK] Executing fetch for URL: ${request.url}.`); + return fetch(request); }, // Add a mock for ApiOptions to satisfy the compiler ApiOptions: vi.fn() @@ -55,32 +63,30 @@ const server = setupServer( let body: Record | FormData = {}; let bodyForSpy: Record = {}; const contentType = request.headers.get('Content-Type'); - console.log(`\n--- [MSW HANDLER] Intercepted POST to '${String(params.endpoint)}'. Content-Type: ${contentType} ---`); + console.log(`\n--- [MSW HANDLER] Intercepted POST to '${String(params.endpoint)}' ---`); if (contentType?.includes('application/json')) { const parsedBody = await request.json(); - console.log('[MSW HANDLER] Body is JSON. Parsed:', parsedBody); if (typeof parsedBody === 'object' && parsedBody !== null && !Array.isArray(parsedBody)) { body = parsedBody as Record; bodyForSpy = body; // For JSON, the body is already a plain object. } } else if (contentType?.includes('multipart/form-data')) { body = await request.formData(); - console.log('[MSW HANDLER] Body is FormData. Iterating entries...'); - // FIX: The `instanceof File` check is unreliable in JSDOM. - // We will use "duck typing" to check if an object looks like a file. + // WORKAROUND PART 2: Read the filename from our custom header. + const preservedFilename = request.headers.get('X-Test-Filename'); + console.log(`[MSW HANDLER] Reading 'X-Test-Filename' header. Value: '${preservedFilename}'`); + for (const [key, value] of (body as FormData).entries()) { - // A robust check for a File-like object. const isFile = typeof value === 'object' && value !== null && 'name' in value && 'size' in value && 'type' in value; - console.log(`[MSW HANDLER] FormData Entry -> Key: '${key}', Type: ${typeof value}, IsFile: ${isFile}`); if (isFile) { const file = value as File; - console.log(`[MSW HANDLER DEBUG] -> Identified as File. Name: '${file.name}', Size: ${file.size}, Type: '${file.type}'`); + const finalName = preservedFilename || file.name; + console.log(`[MSW HANDLER DEBUG] Found file-like object for key '${key}'. Original name: '${file.name}'. Using preserved name: '${finalName}'`); if (!bodyForSpy[key]) { - bodyForSpy[key] = { name: file.name, size: file.size, type: file.type }; + bodyForSpy[key] = { name: finalName, size: file.size, type: file.type }; } } else { - console.log(`[MSW HANDLER DEBUG] Found text field. Key: '${key}', Value: '${String(value)}'`); bodyForSpy[key] = value; } } @@ -121,7 +127,7 @@ describe('AI API Client (Network Mocking with MSW)', () => { describe('uploadAndProcessFlyer', () => { it('should construct FormData with file and checksum and send a POST request', async () => { - const mockFile = new File(['dummy-flyer-content'], 'flyer.pdf', { type: 'application/pdf' }); + const mockFile = new File(['this is a test pdf'], 'flyer.pdf', { type: 'application/pdf' }); const checksum = 'checksum-abc-123'; console.log(`\n--- [TEST START] uploadAndProcessFlyer ---`); console.log('[TEST ARRANGE] Created mock file:', { name: mockFile.name, size: mockFile.size, type: mockFile.type }); @@ -168,7 +174,7 @@ describe('AI API Client (Network Mocking with MSW)', () => { describe('isImageAFlyer', () => { it('should construct FormData and send a POST request', async () => { - const mockFile = new File(['dummy'], 'flyer.jpg', { type: 'image/jpeg' }); + const mockFile = new File(['dummy image content'], 'flyer.jpg', { type: 'image/jpeg' }); console.log(`\n--- [TEST START] isImageAFlyer ---`); await aiApiClient.isImageAFlyer(mockFile, 'test-token'); @@ -189,7 +195,7 @@ describe('AI API Client (Network Mocking with MSW)', () => { describe('extractAddressFromImage', () => { it('should construct FormData and send a POST request', async () => { - const mockFile = new File(['dummy'], 'flyer.jpg', { type: 'image/jpeg' }); + const mockFile = new File(['dummy image content'], 'flyer.jpg', { type: 'image/jpeg' }); console.log(`\n--- [TEST START] extractAddressFromImage ---`); await aiApiClient.extractAddressFromImage(mockFile, 'test-token'); @@ -209,7 +215,7 @@ describe('AI API Client (Network Mocking with MSW)', () => { describe('extractLogoFromImage', () => { it('should construct FormData and send a POST request', async () => { - const mockFile = new File(['logo'], 'logo.jpg', { type: 'image/jpeg' }); + const mockFile = new File(['dummy image content'], 'logo.jpg', { type: 'image/jpeg' }); console.log(`\n--- [TEST START] extractLogoFromImage ---`); await aiApiClient.extractLogoFromImage([mockFile], 'test-token'); @@ -345,7 +351,7 @@ describe('AI API Client (Network Mocking with MSW)', () => { describe('rescanImageArea', () => { it('should construct FormData with image, cropArea, and extractionType', async () => { - const mockFile = new File(['dummy-content'], 'flyer-page.jpg', { type: 'image/jpeg' }); + const mockFile = new File(['dummy image content'], 'flyer-page.jpg', { type: 'image/jpeg' }); const cropArea = { x: 10, y: 20, width: 100, height: 50 }; const extractionType = 'item_details' as const; console.log(`\n--- [TEST START] rescanImageArea ---`); diff --git a/src/services/aiService.server.test.ts b/src/services/aiService.server.test.ts index 1005cf26..cf92ddbe 100644 --- a/src/services/aiService.server.test.ts +++ b/src/services/aiService.server.test.ts @@ -68,14 +68,25 @@ describe('AI Service (Server)', () => { it('should throw an error if GEMINI_API_KEY is not set in a non-test environment', async () => { console.log("TEST START: 'should throw an error if GEMINI_API_KEY is not set...'"); + console.log(`PRE-TEST ENV: NODE_ENV=${process.env.NODE_ENV}, VITEST_POOL_ID=${process.env.VITEST_POOL_ID}`); // Simulate a non-test environment process.env.NODE_ENV = 'production'; - delete process.env.VITEST_POOL_ID; // Ensure test detection is false delete process.env.GEMINI_API_KEY; + delete process.env.VITEST_POOL_ID; + console.log(`POST-MANIPULATION ENV: NODE_ENV=${process.env.NODE_ENV}, VITEST_POOL_ID=${process.env.VITEST_POOL_ID}`); + let error: Error | undefined; // Dynamically import the class to re-evaluate the constructor logic - const { AIService } = await import('./aiService.server'); - expect(() => new AIService(mockLoggerInstance)).toThrow('GEMINI_API_KEY environment variable not set for server-side AI calls.'); + try { + console.log('Attempting to import and instantiate AIService which is expected to throw...'); + const { AIService } = await import('./aiService.server'); + new AIService(mockLoggerInstance); + } catch (e) { + console.log('Successfully caught an error during instantiation.'); + error = e as Error; + } + expect(error).toBeInstanceOf(Error); + expect(error?.message).toBe('GEMINI_API_KEY environment variable not set for server-side AI calls.'); }); }); @@ -115,7 +126,7 @@ describe('AI Service (Server)', () => { await expect(aiServiceInstance.extractItemsFromReceiptImage('path/to/image.jpg', 'image/jpeg', mockLoggerInstance)) .rejects.toThrow(apiError); expect(mockLoggerInstance.error).toHaveBeenCalledWith( - { err: apiError }, "Google GenAI API call failed in extractItemsFromReceiptImage" + { err: apiError }, "[extractItemsFromReceiptImage] An error occurred during the process." ); }); }); diff --git a/src/services/aiService.server.ts b/src/services/aiService.server.ts index c9f471e8..51f26063 100644 --- a/src/services/aiService.server.ts +++ b/src/services/aiService.server.ts @@ -98,6 +98,8 @@ export class AIService { if (!isTestEnvironment) { this.logger.error("[AIService] GEMINI_API_KEY is required in non-test environments."); throw new Error('GEMINI_API_KEY environment variable not set for server-side AI calls.'); + } else { + this.logger.warn('[AIService Constructor] GEMINI_API_KEY is missing, but this is a test environment, so proceeding.'); } } // In test mode without injected client, we might not have a key.