change recommendations to update through doctor

This commit is contained in:
Danel Kungla
2025-10-23 16:48:27 +03:00
parent cd6476821a
commit 9a01c15a76
8 changed files with 205 additions and 147 deletions

View File

@@ -29,7 +29,10 @@ async function UserHomePage() {
const { account } = await loadCurrentUserAccount(); const { account } = await loadCurrentUserAccount();
const api = createUserAnalysesApi(client); const api = createUserAnalysesApi(client);
const userAnalysesApi = createUserAnalysesApi(client);
const bmiThresholds = await api.fetchBmiThresholds(); const bmiThresholds = await api.fetchBmiThresholds();
const analysisResponses = await userAnalysesApi.getAllUserAnalysisResponses();
if (!account) { if (!account) {
redirect('/'); redirect('/');
@@ -54,7 +57,7 @@ async function UserHomePage() {
<Trans i18nKey="dashboard:recommendations.title" /> <Trans i18nKey="dashboard:recommendations.title" />
</h4> </h4>
<div className="mt-4 grid gap-6 sm:grid-cols-3"> <div className="mt-4 grid gap-6 sm:grid-cols-3">
<AIBlocks account={account} /> <AIBlocks account={account} analysisResponses={analysisResponses} />
</div> </div>
</PageBody> </PageBody>
</> </>

View File

@@ -5,27 +5,33 @@ import React, { Suspense } from 'react';
import { AccountWithParams } from '@/packages/features/accounts/src/types/accounts'; import { AccountWithParams } from '@/packages/features/accounts/src/types/accounts';
import { isValidOpenAiEnv } from '../../_lib/server/is-valid-open-ai-env'; import { isValidOpenAiEnv } from '../../_lib/server/is-valid-open-ai-env';
import { loadAnalyses } from '../../_lib/server/load-analyses';
import LifeStyleCard from './life-style-card'; import LifeStyleCard from './life-style-card';
import OrderAnalysesPackageCard from './order-analyses-package-card'; import OrderAnalysesPackageCard from './order-analyses-package-card';
import Recommendations from './recommendations'; import Recommendations from './recommendations';
import RecommendationsSkeleton from './recommendations-skeleton'; import RecommendationsSkeleton from './recommendations-skeleton';
import { AnalysisResponses } from './types';
const AIBlocks = async ({ account }: { account: AccountWithParams }) => { const AIBlocks = async ({
account,
analysisResponses,
}: {
account: AccountWithParams;
analysisResponses?: AnalysisResponses;
}) => {
const isOpenAiAvailable = await isValidOpenAiEnv(); const isOpenAiAvailable = await isValidOpenAiEnv();
if (!isOpenAiAvailable) { if (!isOpenAiAvailable) {
return <OrderAnalysesPackageCard />; return <OrderAnalysesPackageCard />;
} }
if (analysisResponses?.length === 0) {
const { analyses } = await loadAnalyses();
if (analyses.length === 0) {
return ( return (
<> <>
<OrderAnalysesPackageCard /> <OrderAnalysesPackageCard />
<Suspense fallback={<RecommendationsSkeleton amount={1} />}> <Suspense fallback={<RecommendationsSkeleton amount={1} />}>
<LifeStyleCard account={account} /> <LifeStyleCard
account={account}
analysisResponses={analysisResponses}
/>
</Suspense> </Suspense>
</> </>
); );
@@ -33,7 +39,7 @@ const AIBlocks = async ({ account }: { account: AccountWithParams }) => {
return ( return (
<Suspense fallback={<RecommendationsSkeleton />}> <Suspense fallback={<RecommendationsSkeleton />}>
<LifeStyleCard account={account} /> <LifeStyleCard account={account} analysisResponses={analysisResponses} />
<Recommendations account={account} /> <Recommendations account={account} />
</Suspense> </Suspense>
); );

View File

@@ -13,9 +13,16 @@ import { Button } from '@kit/ui/shadcn/button';
import { Card, CardHeader } from '@kit/ui/shadcn/card'; import { Card, CardHeader } from '@kit/ui/shadcn/card';
import { loadLifeStyle } from '../../_lib/server/load-life-style'; import { loadLifeStyle } from '../../_lib/server/load-life-style';
import { AnalysisResponses } from './types';
const LifeStyleCard = async ({ account }: { account: AccountWithParams }) => { const LifeStyleCard = async ({
const data = await loadLifeStyle(account); account,
analysisResponses,
}: {
account: AccountWithParams;
analysisResponses?: AnalysisResponses;
}) => {
const data = await loadLifeStyle(account, analysisResponses);
return ( return (
<Card variant="gradient-success" className="flex flex-col justify-between"> <Card variant="gradient-success" className="flex flex-col justify-between">

View File

@@ -15,7 +15,7 @@ export default async function Recommendations({
}) { }) {
const { analyses, countryCode } = await loadAnalyses(); const { analyses, countryCode } = await loadAnalyses();
const analysisRecommendations = await loadRecommendations(analyses, account); const analysisRecommendations = await loadRecommendations(account);
const orderAnalyses = analyses.filter((analysis) => const orderAnalyses = analyses.filter((analysis) =>
analysisRecommendations.includes(analysis.title), analysisRecommendations.includes(analysis.title),
); );

View File

@@ -1,7 +1,6 @@
'use server'; 'use server';
import { AccountWithParams } from '@/packages/features/accounts/src/types/accounts'; import { AccountWithParams } from '@/packages/features/accounts/src/types/accounts';
import { createUserAnalysesApi } from '@/packages/features/user-analyses/src/server/api';
import { getSupabaseServerClient } from '@/packages/supabase/src/clients/server-client'; import { getSupabaseServerClient } from '@/packages/supabase/src/clients/server-client';
import OpenAI from 'openai'; import OpenAI from 'openai';
@@ -12,8 +11,9 @@ import {
ILifeStyleResponse, ILifeStyleResponse,
PROMPT_NAME, PROMPT_NAME,
} from '../../_components/ai/types'; } from '../../_components/ai/types';
import { OrderAnalysisCard } from '../../_components/order-analyses-cards';
async function getLatestResponseTime(items: AnalysisResponses) { async function getLatestResponseTime(items?: AnalysisResponses) {
if (!items?.length) return null; if (!items?.length) return null;
let latest = null; let latest = null;
@@ -29,8 +29,10 @@ async function getLatestResponseTime(items: AnalysisResponses) {
export async function updateLifeStyle({ export async function updateLifeStyle({
account, account,
analysisResponses,
}: { }: {
account: AccountWithParams; account: AccountWithParams;
analysisResponses?: AnalysisResponses;
}): Promise<ILifeStyleResponse> { }): Promise<ILifeStyleResponse> {
const LIFE_STYLE_PROMPT_ID = process.env.PROMPT_ID_LIFE_STYLE; const LIFE_STYLE_PROMPT_ID = process.env.PROMPT_ID_LIFE_STYLE;
if (!LIFE_STYLE_PROMPT_ID || !account?.personal_code) { if (!LIFE_STYLE_PROMPT_ID || !account?.personal_code) {
@@ -42,13 +44,26 @@ export async function updateLifeStyle({
const openAIClient = new OpenAI(); const openAIClient = new OpenAI();
const supabaseClient = getSupabaseServerClient(); const supabaseClient = getSupabaseServerClient();
const userAnalysesApi = createUserAnalysesApi(supabaseClient);
const analysisResponses = await userAnalysesApi.getAllUserAnalysisResponses();
const { gender, age } = PersonalCode.parsePersonalCode(account.personal_code); const { gender, age } = PersonalCode.parsePersonalCode(account.personal_code);
const weight = account.accountParams?.weight || 'unknown'; const weight = account.accountParams?.weight || 'unknown';
const height = account.accountParams?.height || 'unknown'; const height = account.accountParams?.height || 'unknown';
const isSmoker = !!account.accountParams?.isSmoker; const isSmoker = !!account.accountParams?.isSmoker;
const cholesterol =
analysisResponses
?.find((ar) => ar.analysis_name_lab === 'Kolesterool')
?.response_value.toString() || 'unknown';
const ldl =
analysisResponses
?.find((ar) => ar.analysis_name_lab === 'LDL kolesterool')
?.response_value.toString() || 'unknown';
const hdl =
analysisResponses
?.find((ar) => ar.analysis_name_lab === 'HDL kolesterool')
?.response_value.toString() || 'unknown';
const vitamind =
analysisResponses
?.find((ar) => ar.analysis_name_lab === 'Vitamiin D (25-OH)')
?.response_value.toString() || 'unknown';
const latestResponseTime = await getLatestResponseTime(analysisResponses); const latestResponseTime = await getLatestResponseTime(analysisResponses);
const latestISO = latestResponseTime const latestISO = latestResponseTime
@@ -65,10 +80,10 @@ export async function updateLifeStyle({
age: age.toString(), age: age.toString(),
weight: weight.toString(), weight: weight.toString(),
height: height.toString(), height: height.toString(),
cholesterol: '', cholesterol,
ldl: '', ldl,
hdl: '', hdl,
vitamind: '', vitamind,
is_smoker: isSmoker.toString(), is_smoker: isSmoker.toString(),
}, },
}, },
@@ -85,10 +100,10 @@ export async function updateLifeStyle({
gender: gender.value, gender: gender.value,
age: age.toString(), age: age.toString(),
weight: weight.toString(), weight: weight.toString(),
cholesterol: '', cholesterol,
ldl: '', ldl,
hdl: '', hdl,
vitamind: '', vitamind,
is_smoker: isSmoker.toString(), is_smoker: isSmoker.toString(),
}), }),
latest_data_change: latestISO, latest_data_change: latestISO,
@@ -106,3 +121,92 @@ export async function updateLifeStyle({
}; };
} }
} }
export async function updateRecommendations({
analyses,
analysisResponses,
account,
}: {
analyses: OrderAnalysisCard[];
analysisResponses?: AnalysisResponses;
account: AccountWithParams;
}) {
const RECOMMENDATIONS_PROMPT_IT =
process.env.PROMPT_ID_ANALYSIS_RECOMMENDATIONS;
if (!RECOMMENDATIONS_PROMPT_IT || !account?.personal_code) {
console.error('No prompt ID for analysis recommendations');
return [];
}
const openAIClient = new OpenAI();
const supabaseClient = getSupabaseServerClient();
const { gender, age } = PersonalCode.parsePersonalCode(account.personal_code);
const weight = account.accountParams?.weight || 'unknown';
const formattedAnalysisResponses = analysisResponses?.map(
({
analysis_name_lab,
response_value,
norm_upper,
norm_lower,
norm_status,
}) => ({
name: analysis_name_lab,
value: response_value,
normUpper: norm_upper,
normLower: norm_lower,
normStatus: norm_status,
}),
);
const formattedAnalyses = analyses.map(({ description, title }) => ({
description,
title,
}));
const latestResponseTime = await getLatestResponseTime(analysisResponses);
const latestISO = latestResponseTime
? new Date(latestResponseTime).toISOString()
: new Date('2025').toISOString();
try {
const response = await openAIClient.responses.create({
store: false,
prompt: {
id: RECOMMENDATIONS_PROMPT_IT,
variables: {
analyses: JSON.stringify(formattedAnalyses),
results: JSON.stringify(formattedAnalysisResponses),
gender: gender.value,
age: age.toString(),
weight: weight.toString(),
},
},
});
await supabaseClient
.schema('medreport')
.from('ai_responses')
.insert({
account_id: account.id,
prompt_name: PROMPT_NAME.ANALYSIS_RECOMMENDATIONS,
prompt_id: RECOMMENDATIONS_PROMPT_IT,
input: JSON.stringify({
analyses: formattedAnalyses,
results: formattedAnalysisResponses,
gender,
age,
weight,
}),
latest_data_change: latestISO,
response: response.output_text,
});
const json = JSON.parse(response.output_text);
return json.recommended;
} catch (error) {
console.error('Error getting recommendations: ', error);
return [];
}
}

View File

@@ -3,7 +3,11 @@ import { cache } from 'react';
import { AccountWithParams } from '@/packages/features/accounts/src/types/accounts'; import { AccountWithParams } from '@/packages/features/accounts/src/types/accounts';
import { getSupabaseServerClient } from '@/packages/supabase/src/clients/server-client'; import { getSupabaseServerClient } from '@/packages/supabase/src/clients/server-client';
import { ILifeStyleResponse, PROMPT_NAME } from '../../_components/ai/types'; import {
AnalysisResponses,
ILifeStyleResponse,
PROMPT_NAME,
} from '../../_components/ai/types';
import { updateLifeStyle } from './ai-actions'; import { updateLifeStyle } from './ai-actions';
const failedResponse = { const failedResponse = {
@@ -13,17 +17,12 @@ const failedResponse = {
async function lifeStyleLoader( async function lifeStyleLoader(
account: AccountWithParams, account: AccountWithParams,
analysisResponses?: AnalysisResponses,
): Promise<ILifeStyleResponse> { ): Promise<ILifeStyleResponse> {
if (!account?.personal_code) { if (!account?.personal_code) {
return failedResponse; return failedResponse;
} }
const lifeStylePromptId = process.env.PROMPT_ID_LIFE_STYLE;
if (!lifeStylePromptId) {
return failedResponse;
}
const supabaseClient = getSupabaseServerClient(); const supabaseClient = getSupabaseServerClient();
const { data, error } = await supabaseClient const { data, error } = await supabaseClient
.schema('medreport') .schema('medreport')
@@ -43,7 +42,7 @@ async function lifeStyleLoader(
if (data?.response) { if (data?.response) {
return JSON.parse(data.response as string); return JSON.parse(data.response as string);
} else { } else {
return await updateLifeStyle({ account }); return await updateLifeStyle({ account, analysisResponses });
} }
} }
export const loadLifeStyle = cache(lifeStyleLoader); export const loadLifeStyle = cache(lifeStyleLoader);

View File

@@ -1,141 +1,38 @@
import { cache } from 'react'; import { cache } from 'react';
import { AccountWithParams } from '@/packages/features/accounts/src/types/accounts'; import { AccountWithParams } from '@/packages/features/accounts/src/types/accounts';
import { createUserAnalysesApi } from '@/packages/features/user-analyses/src/server/api';
import { getSupabaseServerClient } from '@/packages/supabase/src/clients/server-client'; import { getSupabaseServerClient } from '@/packages/supabase/src/clients/server-client';
import { Database } from '@/packages/supabase/src/database.types';
import OpenAI from 'openai';
import PersonalCode from '~/lib/utils';
import { PROMPT_NAME } from '../../_components/ai/types'; import { PROMPT_NAME } from '../../_components/ai/types';
import { OrderAnalysisCard } from '../../_components/order-analyses-cards';
export const loadRecommendations = cache(recommendationsLoader); export const loadRecommendations = cache(recommendationsLoader);
type AnalysisResponses =
Database['medreport']['Functions']['get_latest_analysis_response_elements_for_current_user']['Returns'];
const getLatestResponseTime = (items: AnalysisResponses) => {
if (!items?.length) return null;
let latest = null;
for (const it of items) {
const d = new Date(it.response_time);
const t = d.getTime();
if (!Number.isNaN(t) && (latest === null || t > latest.getTime())) {
latest = d;
}
}
return latest;
};
async function recommendationsLoader( async function recommendationsLoader(
analyses: OrderAnalysisCard[],
account: AccountWithParams | null, account: AccountWithParams | null,
): Promise<string[]> { ): Promise<string[]> {
if (!process.env.OPENAI_API_KEY) {
return [];
}
if (!account?.personal_code) { if (!account?.personal_code) {
return []; return [];
} }
const supabaseClient = getSupabaseServerClient(); const supabaseClient = getSupabaseServerClient();
const userAnalysesApi = createUserAnalysesApi(supabaseClient);
const analysisResponses = await userAnalysesApi.getAllUserAnalysisResponses();
const analysesRecommendationsPromptId =
process.env.PROMPT_ID_ANALYSIS_RECOMMENDATIONS;
const latestResponseTime = getLatestResponseTime(analysisResponses);
const latestISO = latestResponseTime
? new Date(latestResponseTime).toISOString()
: new Date('2025').toISOString();
if (!analysesRecommendationsPromptId) { const { data, error } = await supabaseClient
console.error('No prompt ID for analysis recommendations');
return [];
}
const previouslyRecommended = await supabaseClient
.schema('medreport') .schema('medreport')
.from('ai_responses') .from('ai_responses')
.select('*') .select('*')
.eq('account_id', account.id) .eq('account_id', account.id)
.eq('prompt_id', analysesRecommendationsPromptId) .eq('prompt_name', PROMPT_NAME.ANALYSIS_RECOMMENDATIONS)
.eq('latest_data_change', latestISO); .order('latest_data_change', { ascending: false, nullsFirst: false })
.limit(1)
.maybeSingle();
if (previouslyRecommended.data?.[0]?.response) { if (error) {
return JSON.parse(previouslyRecommended.data[0].response as string) console.error('Error fetching AI response from DB: ', error);
.recommended;
}
const openAIClient = new OpenAI();
const { gender, age } = PersonalCode.parsePersonalCode(account.personal_code);
const weight = account.accountParams?.weight || 'unknown';
const formattedAnalysisResponses = analysisResponses.map(
({
analysis_name_lab,
response_value,
norm_upper,
norm_lower,
norm_status,
}) => ({
name: analysis_name_lab,
value: response_value,
normUpper: norm_upper,
normLower: norm_lower,
normStatus: norm_status,
}),
);
const formattedAnalyses = analyses.map(({ description, title }) => ({
description,
title,
}));
let response;
try {
response = await openAIClient.responses.create({
store: false,
prompt: {
id: analysesRecommendationsPromptId,
variables: {
analyses: JSON.stringify(formattedAnalyses),
results: JSON.stringify(formattedAnalysisResponses),
gender: gender.value,
age: age.toString(),
weight: weight.toString(),
},
},
});
} catch (error) {
console.error('Error calling OpenAI: ', error);
return []; return [];
} }
const json = JSON.parse(response.output_text); if (data?.response) {
return JSON.parse(data.response as string).recommended;
try { } else {
await supabaseClient return [];
.schema('medreport')
.from('ai_responses')
.insert({
account_id: account.id,
prompt_name: PROMPT_NAME.ANALYSIS_RECOMMENDATIONS,
prompt_id: analysesRecommendationsPromptId,
input: JSON.stringify({
analyses: formattedAnalyses,
results: formattedAnalysisResponses,
gender,
age,
weight,
}),
latest_data_change: latestISO,
response: response.output_text,
});
} catch (error) {
console.error('Error saving AI response: ', error);
} }
return json.recommended;
} }

View File

@@ -0,0 +1,42 @@
drop function if exists medreport.get_latest_analysis_response_elements_for_current_user(uuid);
create or replace function medreport.get_latest_analysis_response_elements_for_current_user(p_user_id uuid)
returns table (
analysis_name medreport.analysis_response_elements.analysis_name%type,
response_time medreport.analysis_response_elements.response_time%type,
norm_upper medreport.analysis_response_elements.norm_upper%type,
norm_lower medreport.analysis_response_elements.norm_lower%type,
norm_status medreport.analysis_response_elements.norm_status%type,
response_value medreport.analysis_response_elements.response_value%type,
analysis_name_lab medreport.analysis_elements.analysis_name_lab%type
)
language sql
as $$
WITH ranked AS (
SELECT
are.analysis_name,
are.response_time,
are.norm_upper,
are.norm_lower,
are.norm_status,
are.response_value,
ae.analysis_name_lab,
ROW_NUMBER() OVER (
PARTITION BY are.analysis_name
ORDER BY are.response_time DESC, are.id DESC
) AS rn
FROM medreport.analysis_responses ar
JOIN medreport.analysis_response_elements are
ON are.analysis_response_id = ar.id
JOIN medreport.analysis_elements ae
ON are.analysis_element_original_id = ae.analysis_id_original
WHERE ar.user_id = auth.uid()
AND ar.order_status IN ('COMPLETED', 'ON_HOLD')
)
SELECT analysis_name, response_time, norm_upper, norm_lower, norm_status, response_value, analysis_name_lab
FROM ranked
WHERE rn = 1
ORDER BY analysis_name;
$$;
grant execute on function medreport.get_latest_analysis_response_elements_for_current_user(uuid) to authenticated, service_role;