add real data to ai
This commit is contained in:
@@ -1,6 +1,6 @@
|
|||||||
import type { AnalysisResultDetailsMapped } from '@/packages/features/user-analyses/src/types/analysis-results';
|
import type { AnalysisResultDetailsMapped } from '@/packages/features/user-analyses/src/types/analysis-results';
|
||||||
|
|
||||||
export type AnalysisTestResponse = Omit<
|
type AnalysisTestResponse = Omit<
|
||||||
AnalysisResultDetailsMapped,
|
AnalysisResultDetailsMapped,
|
||||||
'order' | 'orderedAnalysisElementIds' | 'summary' | 'elements'
|
'order' | 'orderedAnalysisElementIds' | 'summary' | 'elements'
|
||||||
>;
|
>;
|
||||||
|
|||||||
@@ -4,10 +4,11 @@ import React from 'react';
|
|||||||
|
|
||||||
import { AccountWithParams } from '@/packages/features/accounts/src/types/accounts';
|
import { AccountWithParams } from '@/packages/features/accounts/src/types/accounts';
|
||||||
|
|
||||||
import { analysisResponses } from '../(dashboard)/analysis-results/test/test-responses';
|
import { Trans } from '@kit/ui/makerkit/trans';
|
||||||
|
|
||||||
import { loadAnalyses } from '../_lib/server/load-analyses';
|
import { loadAnalyses } from '../_lib/server/load-analyses';
|
||||||
import { loadRecommendations } from '../_lib/server/load-recommendations';
|
import { loadRecommendations } from '../_lib/server/load-recommendations';
|
||||||
import OrderAnalysesCards, { OrderAnalysisCard } from './order-analyses-cards';
|
import OrderAnalysesCards from './order-analyses-cards';
|
||||||
|
|
||||||
export default async function Recommendations({
|
export default async function Recommendations({
|
||||||
account,
|
account,
|
||||||
@@ -15,25 +16,21 @@ export default async function Recommendations({
|
|||||||
account: AccountWithParams;
|
account: AccountWithParams;
|
||||||
}) {
|
}) {
|
||||||
const { analyses, countryCode } = await loadAnalyses();
|
const { analyses, countryCode } = await loadAnalyses();
|
||||||
const analysisResults = analysisResponses;
|
|
||||||
|
|
||||||
const analysisRecommendations = await loadRecommendations(
|
|
||||||
analysisResults,
|
|
||||||
analyses,
|
|
||||||
account,
|
|
||||||
);
|
|
||||||
|
|
||||||
|
const analysisRecommendations = await loadRecommendations(analyses, account);
|
||||||
const orderAnalyses = analyses.filter((analysis) =>
|
const orderAnalyses = analyses.filter((analysis) =>
|
||||||
analysisRecommendations.includes(analysis.title),
|
analysisRecommendations.includes(analysis.title),
|
||||||
);
|
);
|
||||||
console.log('analysisRecommendations', analysisRecommendations);
|
|
||||||
if (orderAnalyses.length < 1) {
|
if (orderAnalyses.length < 1) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<h4>Medreport soovitab teile</h4>
|
<h4>
|
||||||
|
<Trans i18nKey="dashboard:recommendations.title" />
|
||||||
|
</h4>
|
||||||
<OrderAnalysesCards analyses={orderAnalyses} countryCode={countryCode} />
|
<OrderAnalysesCards analyses={orderAnalyses} countryCode={countryCode} />
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,29 +1,26 @@
|
|||||||
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 { createServerClient } from '@supabase/ssr';
|
import { Database } from '@/packages/supabase/src/database.types';
|
||||||
import OpenAI from 'openai';
|
import OpenAI from 'openai';
|
||||||
|
|
||||||
import PersonalCode from '~/lib/utils';
|
import PersonalCode from '~/lib/utils';
|
||||||
|
|
||||||
import { AnalysisTestResponse } from '../../(dashboard)/analysis-results/test/test-responses';
|
|
||||||
import { OrderAnalysisCard } from '../../_components/order-analyses-cards';
|
import { OrderAnalysisCard } from '../../_components/order-analyses-cards';
|
||||||
|
|
||||||
export const loadRecommendations = cache(recommendationsLoader);
|
export const loadRecommendations = cache(recommendationsLoader);
|
||||||
|
|
||||||
type FormattedAnalysisResponse = {
|
type AnalysisResponses =
|
||||||
value: string;
|
Database['medreport']['Functions']['get_latest_analysis_response_elements_for_current_user']['Returns'];
|
||||||
name: string;
|
|
||||||
responseTime: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
const getLatestResponseTime = (items: FormattedAnalysisResponse[]) => {
|
const getLatestResponseTime = (items: AnalysisResponses) => {
|
||||||
if (!items?.length) return null;
|
if (!items?.length) return null;
|
||||||
|
|
||||||
let latest = null;
|
let latest = null;
|
||||||
for (const it of items) {
|
for (const it of items) {
|
||||||
const d = new Date(it.responseTime);
|
const d = new Date(it.response_time);
|
||||||
const t = d.getTime();
|
const t = d.getTime();
|
||||||
if (!Number.isNaN(t) && (latest === null || t > latest.getTime())) {
|
if (!Number.isNaN(t) && (latest === null || t > latest.getTime())) {
|
||||||
latest = d;
|
latest = d;
|
||||||
@@ -32,45 +29,10 @@ const getLatestResponseTime = (items: FormattedAnalysisResponse[]) => {
|
|||||||
return latest;
|
return latest;
|
||||||
};
|
};
|
||||||
|
|
||||||
const getLatestUniqueAnalysResponses = (
|
|
||||||
analysisResponses: AnalysisTestResponse[],
|
|
||||||
): { name: string; value: string; responseTime: string }[] => {
|
|
||||||
const analysisElements = analysisResponses
|
|
||||||
.map(({ orderedAnalysisElements }) => orderedAnalysisElements)
|
|
||||||
.flat();
|
|
||||||
|
|
||||||
console.log('analysisElements', analysisElements.length);
|
|
||||||
|
|
||||||
const byName = analysisElements.reduce<
|
|
||||||
Record<string, { name: string; value: string; responseTime: string }>
|
|
||||||
>((acc, it) => {
|
|
||||||
const responseTime = it?.results?.responseTime;
|
|
||||||
const responseValue = it?.results?.responseValue;
|
|
||||||
if (!responseTime || !responseValue) return acc;
|
|
||||||
|
|
||||||
const key = it.analysisName;
|
|
||||||
const cur = acc[key];
|
|
||||||
const t = Date.parse(responseTime);
|
|
||||||
const prevT = cur ? Date.parse(cur.responseTime) : -Infinity;
|
|
||||||
|
|
||||||
if (!cur || t > prevT) {
|
|
||||||
acc[key] = {
|
|
||||||
name: key,
|
|
||||||
value: responseValue.toString(),
|
|
||||||
responseTime,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
return acc;
|
|
||||||
}, {});
|
|
||||||
|
|
||||||
return Object.values(byName);
|
|
||||||
};
|
|
||||||
|
|
||||||
async function recommendationsLoader(
|
async function recommendationsLoader(
|
||||||
analysisResponses: AnalysisTestResponse[],
|
|
||||||
analyses: OrderAnalysisCard[],
|
analyses: OrderAnalysisCard[],
|
||||||
account: AccountWithParams | null,
|
account: AccountWithParams | null,
|
||||||
): Promise<any> {
|
): Promise<string[]> {
|
||||||
if (!process.env.OPENAI_API_KEY) {
|
if (!process.env.OPENAI_API_KEY) {
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
@@ -78,15 +40,14 @@ async function recommendationsLoader(
|
|||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
const supabaseClient = getSupabaseServerClient();
|
const supabaseClient = getSupabaseServerClient();
|
||||||
|
const userAnalysesApi = createUserAnalysesApi(supabaseClient);
|
||||||
|
const analysisResponses = await userAnalysesApi.getAllUserAnalysisResponses();
|
||||||
const analysesRecommendationsPromptId =
|
const analysesRecommendationsPromptId =
|
||||||
'pmpt_68ca9c8bfa8c8193b27eadc6496c36440df449ece4f5a8dd';
|
'pmpt_68ca9c8bfa8c8193b27eadc6496c36440df449ece4f5a8dd';
|
||||||
const latestUniqueAnalysResponses =
|
const latestResponseTime = getLatestResponseTime(analysisResponses);
|
||||||
getLatestUniqueAnalysResponses(analysisResponses);
|
|
||||||
const latestResponseTime = getLatestResponseTime(latestUniqueAnalysResponses);
|
|
||||||
const latestISO = latestResponseTime
|
const latestISO = latestResponseTime
|
||||||
? new Date(latestResponseTime).toISOString()
|
? new Date(latestResponseTime).toISOString()
|
||||||
: 'none';
|
: new Date('2025').toISOString();
|
||||||
|
|
||||||
const previouslyRecommended = await supabaseClient
|
const previouslyRecommended = await supabaseClient
|
||||||
.schema('medreport')
|
.schema('medreport')
|
||||||
@@ -97,16 +58,28 @@ async function recommendationsLoader(
|
|||||||
.eq('latest_data_change', latestISO);
|
.eq('latest_data_change', latestISO);
|
||||||
|
|
||||||
if (previouslyRecommended.data?.[0]?.response) {
|
if (previouslyRecommended.data?.[0]?.response) {
|
||||||
return previouslyRecommended.data[0].response;
|
return JSON.parse(previouslyRecommended.data[0].response as string)
|
||||||
|
.recommended;
|
||||||
}
|
}
|
||||||
|
|
||||||
const openAIClient = new OpenAI();
|
const openAIClient = new OpenAI();
|
||||||
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';
|
||||||
console.log('analysisResponses', analysisResponses);
|
|
||||||
console.log('analyises', analyses);
|
const formattedAnalysisResponses = analysisResponses.map(
|
||||||
const formattedAnalysisResponses = latestUniqueAnalysResponses.map(
|
({
|
||||||
({ name, value }) => ({ name, value }),
|
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 }) => ({
|
const formattedAnalyses = analyses.map(({ description, title }) => ({
|
||||||
description,
|
description,
|
||||||
@@ -128,23 +101,28 @@ async function recommendationsLoader(
|
|||||||
});
|
});
|
||||||
|
|
||||||
const json = JSON.parse(response.output_text);
|
const json = JSON.parse(response.output_text);
|
||||||
const updateAiResponse = await supabaseClient
|
|
||||||
.schema('medreport')
|
try {
|
||||||
.from('ai_responses')
|
await supabaseClient
|
||||||
.insert({
|
.schema('medreport')
|
||||||
account_id: account.id,
|
.from('ai_responses')
|
||||||
prompt_name: 'Analysis Recommendations',
|
.insert({
|
||||||
prompt_id: analysesRecommendationsPromptId,
|
account_id: account.id,
|
||||||
input: JSON.stringify({
|
prompt_name: 'Analysis Recommendations',
|
||||||
analyses: formattedAnalyses,
|
prompt_id: analysesRecommendationsPromptId,
|
||||||
results: formattedAnalysisResponses,
|
input: JSON.stringify({
|
||||||
gender,
|
analyses: formattedAnalyses,
|
||||||
age,
|
results: formattedAnalysisResponses,
|
||||||
weight,
|
gender,
|
||||||
}),
|
age,
|
||||||
latest_data_change: latestISO,
|
weight,
|
||||||
response: response.output_text,
|
}),
|
||||||
});
|
latest_data_change: latestISO,
|
||||||
|
response: response.output_text,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Error saving AI response: ', error);
|
||||||
|
}
|
||||||
|
|
||||||
return json.recommended;
|
return json.recommended;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,7 +4,6 @@ import { use } from 'react';
|
|||||||
|
|
||||||
import { CompanyGuard } from '@/packages/features/team-accounts/src/components';
|
import { CompanyGuard } from '@/packages/features/team-accounts/src/components';
|
||||||
import { createTeamAccountsApi } from '@/packages/features/team-accounts/src/server/api';
|
import { createTeamAccountsApi } from '@/packages/features/team-accounts/src/server/api';
|
||||||
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 { PageBody } from '@kit/ui/page';
|
import { PageBody } from '@kit/ui/page';
|
||||||
|
|||||||
@@ -425,6 +425,31 @@ class UserAnalysesApi {
|
|||||||
}
|
}
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async getAllUserAnalysisResponses(): Promise<
|
||||||
|
Database['medreport']['Functions']['get_latest_analysis_response_elements_for_current_user']['Returns']
|
||||||
|
> {
|
||||||
|
const {
|
||||||
|
data: { user },
|
||||||
|
} = await this.client.auth.getUser();
|
||||||
|
|
||||||
|
if (!user) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
const { data, error } = await this.client
|
||||||
|
.schema('medreport')
|
||||||
|
.rpc('get_latest_analysis_response_elements_for_current_user', {
|
||||||
|
p_user_id: user.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
console.error('Error fetching user analysis responses: ', error);
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
|
||||||
|
return data;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export function createUserAnalysesApi(client: SupabaseClient<Database>) {
|
export function createUserAnalysesApi(client: SupabaseClient<Database>) {
|
||||||
|
|||||||
@@ -764,8 +764,8 @@ export type Database = {
|
|||||||
norm_upper?: number | null
|
norm_upper?: number | null
|
||||||
norm_upper_included?: boolean | null
|
norm_upper_included?: boolean | null
|
||||||
original_response_element: Json
|
original_response_element: Json
|
||||||
response_time: string | null
|
response_time?: string | null
|
||||||
response_value: number | null
|
response_value?: number | null
|
||||||
response_value_is_negative?: boolean | null
|
response_value_is_negative?: boolean | null
|
||||||
response_value_is_within_norm?: boolean | null
|
response_value_is_within_norm?: boolean | null
|
||||||
status?: string | null
|
status?: string | null
|
||||||
@@ -1335,6 +1335,7 @@ export type Database = {
|
|||||||
medipost_private_message_id: string | null
|
medipost_private_message_id: string | null
|
||||||
medusa_order_id: string | null
|
medusa_order_id: string | null
|
||||||
response_xml: string | null
|
response_xml: string | null
|
||||||
|
updated_at: string | null
|
||||||
xml: string | null
|
xml: string | null
|
||||||
}
|
}
|
||||||
Insert: {
|
Insert: {
|
||||||
@@ -1347,6 +1348,7 @@ export type Database = {
|
|||||||
medipost_private_message_id?: string | null
|
medipost_private_message_id?: string | null
|
||||||
medusa_order_id?: string | null
|
medusa_order_id?: string | null
|
||||||
response_xml?: string | null
|
response_xml?: string | null
|
||||||
|
updated_at?: string | null
|
||||||
xml?: string | null
|
xml?: string | null
|
||||||
}
|
}
|
||||||
Update: {
|
Update: {
|
||||||
@@ -1359,6 +1361,7 @@ export type Database = {
|
|||||||
medipost_private_message_id?: string | null
|
medipost_private_message_id?: string | null
|
||||||
medusa_order_id?: string | null
|
medusa_order_id?: string | null
|
||||||
response_xml?: string | null
|
response_xml?: string | null
|
||||||
|
updated_at?: string | null
|
||||||
xml?: string | null
|
xml?: string | null
|
||||||
}
|
}
|
||||||
Relationships: []
|
Relationships: []
|
||||||
@@ -2030,6 +2033,18 @@ export type Database = {
|
|||||||
personal_code: string
|
personal_code: string
|
||||||
}[]
|
}[]
|
||||||
}
|
}
|
||||||
|
get_latest_analysis_response_elements_for_current_user: {
|
||||||
|
Args: { p_user_id: string }
|
||||||
|
Returns: {
|
||||||
|
analysis_name: string
|
||||||
|
analysis_name_lab: string
|
||||||
|
norm_lower: number
|
||||||
|
norm_status: number
|
||||||
|
norm_upper: number
|
||||||
|
response_time: string
|
||||||
|
response_value: number
|
||||||
|
}[]
|
||||||
|
}
|
||||||
get_latest_medipost_dispatch_state_for_order: {
|
get_latest_medipost_dispatch_state_for_order: {
|
||||||
Args: { medusa_order_id: string }
|
Args: { medusa_order_id: string }
|
||||||
Returns: {
|
Returns: {
|
||||||
|
|||||||
@@ -18,5 +18,8 @@
|
|||||||
"title": "Order analysis",
|
"title": "Order analysis",
|
||||||
"description": "Select an analysis to get started"
|
"description": "Select an analysis to get started"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"recommendations": {
|
||||||
|
"title": "Medreport recommends"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,5 +18,8 @@
|
|||||||
"title": "Telli analüüs",
|
"title": "Telli analüüs",
|
||||||
"description": "Telli endale sobiv analüüs"
|
"description": "Telli endale sobiv analüüs"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"recommendations": {
|
||||||
|
"title": "Medreport soovitab teile"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,5 +18,8 @@
|
|||||||
"title": "Заказать анализ",
|
"title": "Заказать анализ",
|
||||||
"description": "Закажите подходящий для вас анализ"
|
"description": "Закажите подходящий для вас анализ"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"recommendations": {
|
||||||
|
"title": "Medreport recommends"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,3 +15,45 @@ USING prompt_name::text;
|
|||||||
|
|
||||||
ALTER TABLE medreport.ai_responses
|
ALTER TABLE medreport.ai_responses
|
||||||
ADD CONSTRAINT ai_responses_id_pkey PRIMARY KEY (id);
|
ADD CONSTRAINT ai_responses_id_pkey PRIMARY KEY (id);
|
||||||
|
|
||||||
|
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 = '9ec20b5a-a939-4e5d-9148-6733e36047f3' -- 👈 your user id
|
||||||
|
AND ar.order_status = 'COMPLETED'
|
||||||
|
)
|
||||||
|
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;
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user