From aa17a402199a47e0ea5fc8d999559261afb79b54 Mon Sep 17 00:00:00 2001 From: Jeremy Labrado Date: Fri, 5 Dec 2025 11:37:31 +0100 Subject: [PATCH 1/2] feat: enhance product summary with health goals, religious requirements, and improved UI - Add support for health goals and religious dietary requirements in product analysis - Include detailed nutritional data (calories, carbs, sugars, fats, proteins, fiber, salt) - Integrate allergen tags, product labels, and categories from Open Food Facts API - Improve prompt to conditionally include only user-specified preferences - Modernize UI with card-based design for AI summary and generated images - Add emoji icons and improved typography for better visual hierarchy - Maintain comprehensive translations for all 5 languages (EN, FR, ES, IT, AR) --- README.md | 2 +- lambda/barcode_product_summary/index.js | 147 ++++++++++++------ lambda/recipe_step_by_step/index.js | 2 +- resources/ui/src/assets/i18n/all.ts | 10 ++ .../pages/components/barcode_ingredients.tsx | 6 +- .../components/barcode_product_summary.tsx | 98 +++++++----- 6 files changed, 171 insertions(+), 94 deletions(-) diff --git a/README.md b/README.md index 4b03d6e..d147258 100644 --- a/README.md +++ b/README.md @@ -223,7 +223,7 @@ The output format is a Markdown file to faciliate the display of the recipe on t - **Challenge**: Present the application in multiple languages -- **Solution**: The same prompt is utilized, but the LLM is instructed to generate the output in a specific language, catering to the user's language preference (English/French). +- **Solution**: The same prompt is utilized, but the LLM is instructed to generate the output in a specific language, catering to the user's language preference (English, French, Spanish, Italian, Arabic). The UI includes comprehensive translations for all features including allergen warnings, ingredient descriptions, and nutritional information. **Direct Allergen Detection and Nutritional Analysis** diff --git a/lambda/barcode_product_summary/index.js b/lambda/barcode_product_summary/index.js index e4d6b49..b639ce7 100644 --- a/lambda/barcode_product_summary/index.js +++ b/lambda/barcode_product_summary/index.js @@ -12,56 +12,93 @@ const PRODUCT_TABLE_NAME = process.env.PRODUCT_TABLE_NAME; const PRODUCT_SUMMARY_TABLE_NAME = process.env.PRODUCT_SUMMARY_TABLE_NAME; const MODEL_ID = "anthropic.claude-3-haiku-20240307-v1:0"; const bedrockRuntimeClient = new client_bedrock_runtime_1.BedrockRuntimeClient({ region: process.env.REGION || 'us-east-1' }); -function generateProductSummaryPrompt(userAllergies, userPreference, productIngredients, productName, language) { - return `Human: - You are a nutrition expert with the task to provide recommendations about a specific product for the user based on the user's allergies and preferences. - Your task involves the following steps: +function generateProductSummaryPrompt(userAllergies, userPreference, userHealthGoal, userReligion, productIngredients, productName, productAllergens, productNutriments, productLabels, productCategories, language) { + // Format nutriments for display + let nutrimentInfo = ''; + if (productNutriments && Object.keys(productNutriments).length > 0) { + nutrimentInfo = '\n\n'; + if (productNutriments['energy-kcal_100g']) + nutrimentInfo += `Calories: ${productNutriments['energy-kcal_100g']} kcal\n`; + if (productNutriments['carbohydrates_100g']) + nutrimentInfo += `Carbohydrates: ${productNutriments['carbohydrates_100g']}g\n`; + if (productNutriments['sugars_100g']) + nutrimentInfo += `Sugars: ${productNutriments['sugars_100g']}g\n`; + if (productNutriments['fat_100g']) + nutrimentInfo += `Fat: ${productNutriments['fat_100g']}g\n`; + if (productNutriments['saturated-fat_100g']) + nutrimentInfo += `Saturated Fat: ${productNutriments['saturated-fat_100g']}g\n`; + if (productNutriments['proteins_100g']) + nutrimentInfo += `Protein: ${productNutriments['proteins_100g']}g\n`; + if (productNutriments['fiber_100g']) + nutrimentInfo += `Fiber: ${productNutriments['fiber_100g']}g\n`; + if (productNutriments['salt_100g']) + nutrimentInfo += `Salt: ${productNutriments['salt_100g']}g\n`; + nutrimentInfo += '\n'; + } + // Format allergens - only if user has allergies + let allergenInfo = ''; + if (userAllergies && productAllergens && productAllergens.length > 0) { + allergenInfo = `\n${productAllergens.join(', ')}\n`; + } + // Format labels + let labelInfo = ''; + if (productLabels && productLabels.length > 0) { + labelInfo = `\n${productLabels.join(', ')}\n`; + } + // Format categories + let categoryInfo = ''; + if (productCategories) { + categoryInfo = `\n${productCategories}\n`; + } + // Build instructions based on what user has set + let instructions = `You are a nutrition expert providing recommendations about a specific product. - 1. Use the user's allergy information, if provided, to ensure that the ingredients in the product are suitable for the user. - 2. Use the user's preferences, if provided, to ensure that the user will enjoy the product. Note that the product can contain additives listed in the additives. Make sure these additives are compatible with user allergies and preferences. - 3. Present three benefits and three disadvantages for the product, ensuring that each list consists of precisely three points. - 4. Provide nutritional recommendations for the product based on its ingredients and the user's needs. - - If the user's allergy information or preferences are not provided or are empty, offer general nutritional advice on the product. - - Example: - Chocolate and hazelnut spread - - {{ - Sucre, sirop de glucose, NOISETTES entières torréfiées, matières grasses végétales (palme, karité), beurre de cacao¹, LAIT entier en poudre, PETIT-LAIT filtré en poudre, LAIT écrémé concentré sucré (LAIT écrémé, sucre), sirop de glucose-fructose, pâte de cacao¹, blancs d'ŒUFS en poudre, émulsifiant (lécithines). Peut contenir ARACHIDES, autres FRUITS À COQUE (AMANDES, NOIX DE CAJOU, NOIX DE PECAN) et SOJA. ¹Bilan massique certifié Rainforest Alliance. www.ra.org/fr. - }} - - - I don't like chocolate - - Response: - - - - Although Nutella contains a small amount of calcium and iron, it's not very nutritious and high in sugar, calories and fat. - - - - {{benefit}} - - - {{disadvantage}} - - + Your task: + `; + if (userAllergies) { + instructions += `1. CRITICAL: Check if any product allergens match the user's allergies (${userAllergies}). If there is a match, prominently warn the user.\n`; + } + if (userPreference) { + instructions += `${userAllergies ? '2' : '1'}. Check if product labels match dietary preferences (${userPreference}). Use labels for direct matching, or analyze categories and ingredients.\n`; + } + if (userHealthGoal) { + instructions += `${(userAllergies ? 1 : 0) + (userPreference ? 1 : 0) + 1}. Use nutritional data to assess if the product aligns with the health goal: ${userHealthGoal}.\n`; + } + if (userReligion) { + instructions += `${(userAllergies ? 1 : 0) + (userPreference ? 1 : 0) + (userHealthGoal ? 1 : 0) + 1}. Check if product labels match religious requirement: ${userReligion}.\n`; + } + instructions += `- Present three nutritional benefits and three nutritional disadvantages for the product based on actual nutritionalvalues. + If the user's information is not provided or is empty, offer general nutritional advice based on the product's nutritional data. + IMPORTANT: Only mention allergens, dietary preferences, health goals, or religious requirements if the user has specified them. Do not discuss aspects the user hasn't set.`; + let userContext = ''; + if (userAllergies) + userContext += `\n${userAllergies}`; + if (userHealthGoal) + userContext += `\n${userHealthGoal}`; + if (userPreference) + userContext += `\n${userPreference}`; + if (userReligion) + userContext += `\n${userReligion}`; + return `Human: + ${instructions} - Provide recommendation for the following product - ${productName} - - ${productIngredients} - - ${userAllergies} - ${userPreference} + Provide recommendation for the following product: + ${productName} + ${productIngredients} + ${allergenInfo} + ${labelInfo} + ${categoryInfo} + ${nutrimentInfo} + + For the user: + ${userContext} + Provide the response in the third person, in ${language}, skip the preambule, disregard any content at the end and provide only the response in this Markdown format: markdown - Describe potential_health_issues, preference_matter and recommendation here combines in one single short paragraph + Describe allergen warnings (if any), dietary label compatibility, religious requirement compatibility, health goal compatibility, dietary preference compatibility, and recommendation here combined in one single short paragraph #### Benefits title here - Describe benefits here @@ -102,7 +139,7 @@ function calculateHash(productCode, userAllergies, userPreferenceData, language) * * @param productCode - The code of the product to retrieve information for. * @param language - The language for the product information. - * @returns A tuple containing product name, ingredients, and additives if the product is found in the database; otherwise, returns [null, null, null]. + * @returns A tuple containing product name, ingredients, additives, allergens, and nutriments if the product is found in the database; otherwise, returns [null, null, null, null, null]. */ async function getProductFromDb(productCode, language) { try { @@ -116,15 +153,23 @@ async function getProductFromDb(productCode, language) { // Check if the item exists if (Item) { const item = (0, util_dynamodb_1.unmarshall)(Item); - return [item.product_name || null, item.ingredients || null, item.additives || null]; + return [ + item.product_name || null, + item.ingredients || null, + item.additives || null, + item.allergens_tags || null, + item.nutriments || null, + item.labels_tags || null, + item.categories || null + ]; } else { - return [null, null, null]; + return [null, null, null, null, null, null, null]; } } catch (e) { console.error('Error while getting the Product from database', e); - return [null, null, null]; + return [null, null, null, null, null, null, null]; } } async function getProductSummary(productCode, paramsHash) { @@ -249,9 +294,11 @@ async function messageHandler(event, responseStream) { const language = body.language; const userPreferenceKeys = Object.keys(body.preferences).filter(key => body.preferences[key]); const userAllergiesKeys = Object.keys(body.allergies).filter(key => body.allergies[key]); + const userHealthGoal = body.healthGoal || ''; + const userReligion = body.religion || ''; const userPreferenceString = userPreferenceKeys.join(', '); const userAllergiesString = userAllergiesKeys.join(', '); - const [productName, productIngredients, productAdditives] = await getProductFromDb(productCode, language); + const [productName, productIngredients, productAdditives, productAllergens, productNutriments, productLabels, productCategories] = await getProductFromDb(productCode, language); if (productName && productIngredients) { logger.info("Product found"); } @@ -265,7 +312,7 @@ async function messageHandler(event, responseStream) { logger.info("Product Summary not found in the database"); const ingredientKeys = Object.keys(productIngredients); const ingredientsString = ingredientKeys.join(', '); - const promptText = generateProductSummaryPrompt(userAllergiesString, userPreferenceString, ingredientsString, productName, language); + const promptText = generateProductSummaryPrompt(userAllergiesString, userPreferenceString, userHealthGoal, userReligion, ingredientsString, productName, productAllergens || [], productNutriments || {}, productLabels || [], productCategories || '', language); productSummary = await generateSummary(promptText, responseStream); await putProductSummaryToDynamoDB(productCode, hashValue, productSummary); } @@ -280,4 +327,4 @@ async function messageHandler(event, responseStream) { responseStream.end(); } exports.handler = awslambda.streamifyResponse(messageHandler); -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import { DynamoDBClient, GetItemCommand, PutItemCommand } from \"@aws-sdk/client-dynamodb\";\nimport { unmarshall } from \"@aws-sdk/util-dynamodb\";\nimport { Tracer } from \"@aws-lambda-powertools/tracer\";\nimport { Logger } from \"@aws-lambda-powertools/logger\";\nimport { APIGatewayProxyEventV2, Handler, Context } from 'aws-lambda';\nimport { createHash } from 'crypto';\nimport { BedrockRuntimeClient, InvokeModelWithResponseStreamCommand } from \"@aws-sdk/client-bedrock-runtime\";\n\nconst logger = new Logger();\nconst dynamodb = new DynamoDBClient({});\n\nconst PRODUCT_TABLE_NAME = process.env.PRODUCT_TABLE_NAME\nconst PRODUCT_SUMMARY_TABLE_NAME = process.env.PRODUCT_SUMMARY_TABLE_NAME\nconst MODEL_ID = \"anthropic.claude-3-haiku-20240307-v1:0\"\n\n\n\nconst bedrockRuntimeClient = new BedrockRuntimeClient({ region: process.env.REGION || 'us-east-1' });\n\n\ndeclare global {\n    namespace awslambda {\n      function streamifyResponse(\n        f: (\n          event: APIGatewayProxyEventV2,\n          responseStream: NodeJS.WritableStream,\n          context: Context\n        ) => Promise<void>\n      ): Handler;\n    }\n}\n\n\n\ninterface ProductItem {\n    product_code: string;\n    language: string;\n    product_name?: string;\n    ingredients?: string;\n    additives?: string;\n}\n\ninterface ProductSummaryItem {\n    product_code: string;\n    params_hash: string;\n    summary: string;\n}\n\ninterface SummaryData {\n    recommendations: string[];\n    benefits: string[];\n    disadvantages: string[];\n  }\n\n\nfunction generateProductSummaryPrompt(\n    userAllergies: string,\n    userPreference: string,\n    productIngredients: string,\n    productName: string,\n    language: string\n    ): string {\n    return `Human:\n          You are a nutrition expert with the task to provide recommendations about a specific product for the user based on the user's allergies and preferences. \n          Your task involves the following steps:\n\n          1. Use the user's allergy information, if provided, to ensure that the ingredients in the product are suitable for the user.\n          2. Use the user's preferences, if provided, to ensure that the user will enjoy the product. Note that the product can contain additives listed in the additives. Make sure these additives are compatible with user allergies and preferences.\n          3. Present three benefits and three disadvantages for the product, ensuring that each list consists of precisely three points.\n          4. Provide nutritional recommendations for the product based on its ingredients and the user's needs.\n  \n          If the user's allergy information or preferences are not provided or are empty, offer general nutritional advice on the product.\n  \n          Example:\n          <product_name>Chocolate and hazelnut spread</product_name>\n          <product_ingredients>\n          {{\n              Sucre, sirop de glucose, NOISETTES entières torréfiées, matières grasses végétales (palme, karité), beurre de cacao¹, LAIT entier en poudre, PETIT-LAIT filtré en poudre, LAIT écrémé concentré sucré (LAIT écrémé, sucre), sirop de glucose-fructose, pâte de cacao¹, blancs d'ŒUFS en poudre, émulsifiant (lécithines). Peut contenir ARACHIDES, autres FRUITS À COQUE (AMANDES, NOIX DE CAJOU, NOIX DE PECAN) et SOJA. ¹Bilan massique certifié Rainforest Alliance. www.ra.org/fr.\n          }}\n          </product_ingredients>\n          <user_allergies></user_allergies>\n          <user_preferences>I don't like chocolate</user_preferences>\n          </example>\n          Response: \n          <data>\n              <recommendations>\n                  <recommendation>\n                  Although Nutella contains a small amount of calcium and iron, it's not very nutritious and high in sugar, calories and fat.\n                  </recommendation>\n              </recommendations>\n              <benefits>\n                  <benefit>{{benefit}}</benefit>\n              </benefits>\n              <disadvantages>\n                  <disadvantage>{{disadvantage}}</disadvantage>\n              </disadvantages>                 \n          </data>\n  \n          Provide recommendation for the following product\n          <product_name>${productName}</product_name>\n          <product_ingredients>\n          ${productIngredients}\n          </product_ingredients>\n          <user_allergies>${userAllergies}</user_allergies>\n          <user_preferences>${userPreference}</user_preferences>\n          Provide the response in the third person, in ${language}, skip the preambule, disregard any content at the end and provide only the response in this Markdown format:\n\n\n        markdown\n\n        Describe potential_health_issues, preference_matter and recommendation here combines in one single short paragraph\n\n        #### Benefits title here\n        - Describe benefits here\n\n        #### Disadvantages title here\n        - Describe disadvantages here\n          \n          Assistant:\n          `;\n}\n\nfunction generateCombinedString(obj: { [key: string]: any }): string {\n    const concatenatedString = Object.keys(obj).join('');\n    return concatenatedString;\n}\n\n\n\nfunction calculateHash(\n    productCode: string,\n    userAllergies: any,\n    userPreferenceData: any,\n    language: string\n    ): string {\n    /**\n     * Calculates a SHA-256 hash based on various input data.\n     *\n     * @param userAllergies - A string containing user allergies data.\n     * @param userPreferenceData - A string containing user preference data.\n     * @param productIngredients - A string containing product ingredients data.\n     * @param productName - The name of the product.\n     * @param language - The language.\n     * @param productAdditives - A string containing product additives data.\n     * @returns The SHA-256 hash value calculated based on the concatenated string representations of the input data.\n     */\n\n    // Convert dictionaries to JSON strings\n    const userAllergiesStr = generateCombinedString(userAllergies);//JSON.stringify(userAllergies);\n    const userPreferenceDataStr = generateCombinedString(userPreferenceData);\n    \n    // Concatenate the string representations of the variables\n    const concatenatedString = `${productCode}${userAllergiesStr}${userPreferenceDataStr}${language}`;\n    // Calculate the hash\n    const hashedValue = createHash('sha256').update(concatenatedString).digest('hex');\n    \n    return hashedValue;\n}\n\n/**\n * Retrieves product information from the database using the provided product code.\n *\n * @param productCode - The code of the product to retrieve information for.\n * @param language - The language for the product information.\n * @returns A tuple containing product name, ingredients, and additives if the product is found in the database; otherwise, returns [null, null, null].\n */\nasync function getProductFromDb(productCode: string, language: string): Promise<[string | null, string | null, string | null]> {\n\n    try {\n        const { Item  = {} } = await dynamodb.send(new GetItemCommand({\n            TableName: PRODUCT_TABLE_NAME,\n            Key: {\n                product_code: { S: productCode },\n                language: { S: language }\n            }\n        }));\n        // Check if the item exists\n        if (Item) {\n            const item = unmarshall(Item) as ProductItem;\n            return [item.product_name || null, item.ingredients || null, item.additives || null];\n        } else {\n            return [null, null, null];\n        }\n    } catch (e) {\n        console.error('Error while getting the Product from database', e);\n        return [null, null, null];\n    }\n}\n\nasync function getProductSummary(productCode: string, paramsHash: string): Promise<string | null> {\n    /**\n     * Retrieves the summary of a product from the database using the product code and parameters hash.\n     *\n     * @param productCode - The code of the product.\n     * @param paramsHash - The hash value representing parameters.\n     * @returns The summary of the product if found in the database; otherwise, returns null.\n     */\n  \n    const { Item  = {} } = await dynamodb.send(new GetItemCommand({\n        TableName: PRODUCT_SUMMARY_TABLE_NAME,\n        Key: {\n            product_code: { S: productCode },\n            params_hash: { S: paramsHash }\n        }\n    }));\n  \n    if (Item) {\n      const item = unmarshall(Item) as ProductSummaryItem;\n      return item.summary;\n    } else {\n      return null;\n    }\n}\n\nasync function generateSummary(promptText, responseStream) {\n\n    const payload = {\n        messages: [\n            {\n                role: \"user\",\n                content: [\n                    {\n                        \"type\": \"text\",\n                        \"text\": promptText\n                    }\n                ]\n            }\n        ],\n        max_tokens: 500,\n        temperature: 0.5,\n        anthropic_version: \"bedrock-2023-05-31\"\n      };\n    const params = {\n        modelId: MODEL_ID,\n        contentType: \"application/json\",\n        accept: \"application/json\",\n        body: JSON.stringify(payload),\n    };\n    let completion = '';\n    try {\n        try {\n            const command = new InvokeModelWithResponseStreamCommand(params);\n            const response = await bedrockRuntimeClient.send(command);\n            const events = response.body;\n            for await (const event of events || []) {\n                // Check the top-level field to determine which event this is.\n                if (event.chunk) {\n                  const decoded_event = JSON.parse(\n                    new TextDecoder().decode(event.chunk.bytes),\n                  );\n                  if (decoded_event.type  === 'content_block_delta' && decoded_event.delta.type === 'text_delta'){\n                    responseStream.write(decoded_event.delta.text)\n                    completion += decoded_event.delta.text;\n                  }\n                } else {\n                  logger.error(`event = ${event}`)\n                }\n              }\n            \n              logger.info('Stream ended!')\n        } catch (err) {\n            // handle error\n            logger.error(err as any);\n        }\n    }\n    catch (e) {\n        logger.error(`Error while generating summary: ${e}`);\n        completion = \"Error while generating summary\";\n    }\n    return completion;\n}\n\nasync function simulateSummaryStreaming(content: string, responseStream): Promise<void> {\n   \n    const chunks = [];\n    let remainingContent = content;\n\n    // Loop until all content is split into chunks\n    while (remainingContent.length > 0) {\n        // Generate a random chunk size between 1 and 10\n        const chunkSize = Math.floor(Math.random() * 10) + 1;\n\n        // Take a chunk of content with the generated chunk size\n        const chunk = remainingContent.slice(0, chunkSize);\n\n        // Add the chunk to the array\n        chunks.push(chunk);\n\n        // Remove the taken chunk from the remaining content\n        remainingContent = remainingContent.slice(chunkSize);\n    }\n\n    // Simulate streaming by emitting each chunk with a delay\n    for (const chunk of chunks) {\n        await new Promise(resolve => setTimeout(resolve, 50)); // Simulate delay\n        responseStream.write(chunk)\n\n    }\n}\n\n\n\n\nasync function putProductSummaryToDynamoDB(product_code: string, params_hash: string, summary: string) {\n    try {\n        await dynamodb.send(new PutItemCommand({\n            TableName: PRODUCT_SUMMARY_TABLE_NAME,\n            Item: {\n                product_code: { S: product_code },\n                params_hash: { S: params_hash },\n                summary: { S: summary }\n            }\n        }));\n        logger.debug(\"Summary saved into database\");\n    } catch (error) {\n        console.error(\"Error:\", error);\n    }\n}\n\nasync function messageHandler (event, responseStream) {\n\n    try {\n        logger.info(event as any);\n\n        const body = event.body ? JSON.parse(event.body) : {};\n        const productCode = body.productCode;\n        const language = body.language;\n\n        const userPreferenceKeys = Object.keys(body.preferences).filter(key => body.preferences[key]);\n        const userAllergiesKeys = Object.keys(body.allergies).filter(key => body.allergies[key]);\n\n        const userPreferenceString = userPreferenceKeys.join(', ');\n        const userAllergiesString = userAllergiesKeys.join(', ');\n\n\n        const [productName, productIngredients, productAdditives] = await getProductFromDb(productCode, language);\n        if (productName && productIngredients) {\n            logger.info(\"Product found\");\n\n\n        } else {\n            logger.error(\"Product not found in the database\");\n            throw new Error('Product not found in the database');\n        }\n\n        const hashValue = calculateHash(productCode, userAllergiesString, userPreferenceString, language);\n\n        let productSummary = await getProductSummary(productCode, hashValue);\n        if (!productSummary) {        \n            logger.info(\"Product Summary not found in the database\");\n            const ingredientKeys = Object.keys(productIngredients);\n            const ingredientsString = ingredientKeys.join(', ');\n\n            const promptText = generateProductSummaryPrompt(\n                userAllergiesString,\n                userPreferenceString,\n                ingredientsString,\n                productName,\n                language!\n            );\n            productSummary = await generateSummary(promptText, responseStream);\n            await putProductSummaryToDynamoDB(productCode, hashValue, productSummary);\n        }\n        else {\n            await simulateSummaryStreaming(productSummary, responseStream)\n\n        }\n        logger.info(`Product Summary: ${productSummary}`);\n    } catch (error) {\n        console.error(\"Error:\", error);\n    }\n    responseStream.end();\n}\n\nexport const handler = awslambda.streamifyResponse(messageHandler);"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":";;;AAAA,8DAA0F;AAC1F,0DAAoD;AAEpD,0DAAuD;AAEvD,mCAAoC;AACpC,4EAA6G;AAE7G,MAAM,MAAM,GAAG,IAAI,eAAM,EAAE,CAAC;AAC5B,MAAM,QAAQ,GAAG,IAAI,gCAAc,CAAC,EAAE,CAAC,CAAC;AAExC,MAAM,kBAAkB,GAAG,OAAO,CAAC,GAAG,CAAC,kBAAkB,CAAA;AACzD,MAAM,0BAA0B,GAAG,OAAO,CAAC,GAAG,CAAC,0BAA0B,CAAA;AACzE,MAAM,QAAQ,GAAG,wCAAwC,CAAA;AAIzD,MAAM,oBAAoB,GAAG,IAAI,6CAAoB,CAAC,EAAE,MAAM,EAAE,OAAO,CAAC,GAAG,CAAC,MAAM,IAAI,WAAW,EAAE,CAAC,CAAC;AA0CrG,SAAS,4BAA4B,CACjC,aAAqB,EACrB,cAAsB,EACtB,cAAsB,EACtB,YAAoB,EACpB,kBAA0B,EAC1B,WAAmB,EACnB,gBAA0B,EAC1B,iBAAsB,EACtB,aAAuB,EACvB,iBAAyB,EACzB,QAAgB;IAGhB,gCAAgC;IAChC,IAAI,aAAa,GAAG,EAAE,CAAC;IACvB,IAAI,iBAAiB,IAAI,MAAM,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,MAAM,GAAG,CAAC,EAAE;QAChE,aAAa,GAAG,0BAA0B,CAAC;QAC3C,IAAI,iBAAiB,CAAC,kBAAkB,CAAC;YAAE,aAAa,IAAI,aAAa,iBAAiB,CAAC,kBAAkB,CAAC,SAAS,CAAC;QACxH,IAAI,iBAAiB,CAAC,oBAAoB,CAAC;YAAE,aAAa,IAAI,kBAAkB,iBAAiB,CAAC,oBAAoB,CAAC,KAAK,CAAC;QAC7H,IAAI,iBAAiB,CAAC,aAAa,CAAC;YAAE,aAAa,IAAI,WAAW,iBAAiB,CAAC,aAAa,CAAC,KAAK,CAAC;QACxG,IAAI,iBAAiB,CAAC,UAAU,CAAC;YAAE,aAAa,IAAI,QAAQ,iBAAiB,CAAC,UAAU,CAAC,KAAK,CAAC;QAC/F,IAAI,iBAAiB,CAAC,oBAAoB,CAAC;YAAE,aAAa,IAAI,kBAAkB,iBAAiB,CAAC,oBAAoB,CAAC,KAAK,CAAC;QAC7H,IAAI,iBAAiB,CAAC,eAAe,CAAC;YAAE,aAAa,IAAI,YAAY,iBAAiB,CAAC,eAAe,CAAC,KAAK,CAAC;QAC7G,IAAI,iBAAiB,CAAC,YAAY,CAAC;YAAE,aAAa,IAAI,UAAU,iBAAiB,CAAC,YAAY,CAAC,KAAK,CAAC;QACrG,IAAI,iBAAiB,CAAC,WAAW,CAAC;YAAE,aAAa,IAAI,SAAS,iBAAiB,CAAC,WAAW,CAAC,KAAK,CAAC;QAClG,aAAa,IAAI,yBAAyB,CAAC;KAC9C;IAED,gDAAgD;IAChD,IAAI,YAAY,GAAG,EAAE,CAAC;IACtB,IAAI,aAAa,IAAI,gBAAgB,IAAI,gBAAgB,CAAC,MAAM,GAAG,CAAC,EAAE;QAClE,YAAY,GAAG,wBAAwB,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,wBAAwB,CAAC;KAC9F;IAED,gBAAgB;IAChB,IAAI,SAAS,GAAG,EAAE,CAAC;IACnB,IAAI,aAAa,IAAI,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE;QAC3C,SAAS,GAAG,qBAAqB,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,qBAAqB,CAAC;KAClF;IAED,oBAAoB;IACpB,IAAI,YAAY,GAAG,EAAE,CAAC;IACtB,IAAI,iBAAiB,EAAE;QACnB,YAAY,GAAG,yBAAyB,iBAAiB,yBAAyB,CAAC;KACtF;IAED,gDAAgD;IAChD,IAAI,YAAY,GAAG;;;KAGlB,CAAC;IAEF,IAAI,aAAa,EAAE;QACf,YAAY,IAAI,2EAA2E,aAAa,sDAAsD,CAAC;KAClK;IAED,IAAI,cAAc,EAAE;QAChB,YAAY,IAAI,GAAG,aAAa,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,GAAG,wDAAwD,cAAc,6EAA6E,CAAC;KACnM;IAED,IAAI,cAAc,EAAE;QAChB,YAAY,IAAI,GAAG,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,gFAAgF,cAAc,KAAK,CAAC;KAChL;IAED,IAAI,YAAY,EAAE;QACd,YAAY,IAAI,GAAG,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,0DAA0D,YAAY,KAAK,CAAC;KACnL;IAED,YAAY,IAAI;;gLAE4J,CAAC;IAE7K,IAAI,WAAW,GAAG,EAAE,CAAC;IACrB,IAAI,aAAa;QAAE,WAAW,IAAI,qBAAqB,aAAa,mBAAmB,CAAC;IACxF,IAAI,cAAc;QAAE,WAAW,IAAI,uBAAuB,cAAc,qBAAqB,CAAC;IAC9F,IAAI,cAAc;QAAE,WAAW,IAAI,+BAA+B,cAAc,6BAA6B,CAAC;IAC9G,IAAI,YAAY;QAAE,WAAW,IAAI,iCAAiC,YAAY,+BAA+B,CAAC;IAE9G,OAAO;YACC,YAAY;;;4BAGI,WAAW;mCACJ,kBAAkB;4BACzB,YAAY;yBACf,SAAS;4BACN,YAAY;6BACX,aAAa;;;cAG5B,WAAW;;yDAEgC,QAAQ;;;;;;;;;;;;;;WActD,CAAC;AACZ,CAAC;AAED,SAAS,sBAAsB,CAAC,GAA2B;IACvD,MAAM,kBAAkB,GAAG,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACrD,OAAO,kBAAkB,CAAC;AAC9B,CAAC;AAID,SAAS,aAAa,CAClB,WAAmB,EACnB,aAAkB,EAClB,kBAAuB,EACvB,QAAgB;IAEhB;;;;;;;;;;OAUG;IAEH,uCAAuC;IACvC,MAAM,gBAAgB,GAAG,sBAAsB,CAAC,aAAa,CAAC,CAAC,CAAA,gCAAgC;IAC/F,MAAM,qBAAqB,GAAG,sBAAsB,CAAC,kBAAkB,CAAC,CAAC;IAEzE,0DAA0D;IAC1D,MAAM,kBAAkB,GAAG,GAAG,WAAW,GAAG,gBAAgB,GAAG,qBAAqB,GAAG,QAAQ,EAAE,CAAC;IAClG,qBAAqB;IACrB,MAAM,WAAW,GAAG,IAAA,mBAAU,EAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,kBAAkB,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;IAElF,OAAO,WAAW,CAAC;AACvB,CAAC;AAED;;;;;;GAMG;AACH,KAAK,UAAU,gBAAgB,CAAC,WAAmB,EAAE,QAAgB;IAEjE,IAAI;QACA,MAAM,EAAE,IAAI,GAAI,EAAE,EAAE,GAAG,MAAM,QAAQ,CAAC,IAAI,CAAC,IAAI,gCAAc,CAAC;YAC1D,SAAS,EAAE,kBAAkB;YAC7B,GAAG,EAAE;gBACD,YAAY,EAAE,EAAE,CAAC,EAAE,WAAW,EAAE;gBAChC,QAAQ,EAAE,EAAE,CAAC,EAAE,QAAQ,EAAE;aAC5B;SACJ,CAAC,CAAC,CAAC;QACJ,2BAA2B;QAC3B,IAAI,IAAI,EAAE;YACN,MAAM,IAAI,GAAG,IAAA,0BAAU,EAAC,IAAI,CAAgB,CAAC;YAC7C,OAAO;gBACH,IAAI,CAAC,YAAY,IAAI,IAAI;gBACzB,IAAI,CAAC,WAAW,IAAI,IAAI;gBACxB,IAAI,CAAC,SAAS,IAAI,IAAI;gBACtB,IAAI,CAAC,cAAc,IAAI,IAAI;gBAC3B,IAAI,CAAC,UAAU,IAAI,IAAI;gBACvB,IAAI,CAAC,WAAW,IAAI,IAAI;gBACxB,IAAI,CAAC,UAAU,IAAI,IAAI;aAC1B,CAAC;SACL;aAAM;YACH,OAAO,CAAC,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;SACrD;KACJ;IAAC,OAAO,CAAC,EAAE;QACR,OAAO,CAAC,KAAK,CAAC,+CAA+C,EAAE,CAAC,CAAC,CAAC;QAClE,OAAO,CAAC,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;KACrD;AACL,CAAC;AAED,KAAK,UAAU,iBAAiB,CAAC,WAAmB,EAAE,UAAkB;IACpE;;;;;;OAMG;IAEH,MAAM,EAAE,IAAI,GAAI,EAAE,EAAE,GAAG,MAAM,QAAQ,CAAC,IAAI,CAAC,IAAI,gCAAc,CAAC;QAC1D,SAAS,EAAE,0BAA0B;QACrC,GAAG,EAAE;YACD,YAAY,EAAE,EAAE,CAAC,EAAE,WAAW,EAAE;YAChC,WAAW,EAAE,EAAE,CAAC,EAAE,UAAU,EAAE;SACjC;KACJ,CAAC,CAAC,CAAC;IAEJ,IAAI,IAAI,EAAE;QACR,MAAM,IAAI,GAAG,IAAA,0BAAU,EAAC,IAAI,CAAuB,CAAC;QACpD,OAAO,IAAI,CAAC,OAAO,CAAC;KACrB;SAAM;QACL,OAAO,IAAI,CAAC;KACb;AACL,CAAC;AAED,KAAK,UAAU,eAAe,CAAC,UAAkB,EAAE,cAAqC;IAEpF,MAAM,OAAO,GAAG;QACZ,QAAQ,EAAE;YACN;gBACI,IAAI,EAAE,MAAM;gBACZ,OAAO,EAAE;oBACL;wBACI,MAAM,EAAE,MAAM;wBACd,MAAM,EAAE,UAAU;qBACrB;iBACJ;aACJ;SACJ;QACD,UAAU,EAAE,GAAG;QACf,WAAW,EAAE,GAAG;QAChB,iBAAiB,EAAE,oBAAoB;KACxC,CAAC;IACJ,MAAM,MAAM,GAAG;QACX,OAAO,EAAE,QAAQ;QACjB,WAAW,EAAE,kBAAkB;QAC/B,MAAM,EAAE,kBAAkB;QAC1B,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC;KAChC,CAAC;IACF,IAAI,UAAU,GAAG,EAAE,CAAC;IACpB,IAAI;QACA,IAAI;YACA,MAAM,OAAO,GAAG,IAAI,6DAAoC,CAAC,MAAM,CAAC,CAAC;YACjE,MAAM,QAAQ,GAAG,MAAM,oBAAoB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YAC1D,MAAM,MAAM,GAAG,QAAQ,CAAC,IAAI,CAAC;YAC7B,IAAI,KAAK,EAAE,MAAM,KAAK,IAAI,MAAM,IAAI,EAAE,EAAE;gBACpC,8DAA8D;gBAC9D,IAAI,KAAK,CAAC,KAAK,EAAE;oBACf,MAAM,aAAa,GAAG,IAAI,CAAC,KAAK,CAC9B,IAAI,WAAW,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,KAAK,CAAC,CAC5C,CAAC;oBACF,IAAI,aAAa,CAAC,IAAI,KAAM,qBAAqB,IAAI,aAAa,CAAC,KAAK,CAAC,IAAI,KAAK,YAAY,EAAC;wBAC7F,cAAc,CAAC,KAAK,CAAC,aAAa,CAAC,KAAK,CAAC,IAAI,CAAC,CAAA;wBAC9C,UAAU,IAAI,aAAa,CAAC,KAAK,CAAC,IAAI,CAAC;qBACxC;iBACF;qBAAM;oBACL,MAAM,CAAC,KAAK,CAAC,WAAW,KAAK,EAAE,CAAC,CAAA;iBACjC;aACF;YAED,MAAM,CAAC,IAAI,CAAC,eAAe,CAAC,CAAA;SACjC;QAAC,OAAO,GAAG,EAAE;YACV,eAAe;YACf,MAAM,CAAC,KAAK,CAAC,GAAU,CAAC,CAAC;SAC5B;KACJ;IACD,OAAO,CAAC,EAAE;QACN,MAAM,CAAC,KAAK,CAAC,mCAAmC,CAAC,EAAE,CAAC,CAAC;QACrD,UAAU,GAAG,gCAAgC,CAAC;KACjD;IACD,OAAO,UAAU,CAAC;AACtB,CAAC;AAED,KAAK,UAAU,wBAAwB,CAAC,OAAe,EAAE,cAAqC;IAE1F,MAAM,MAAM,GAAG,EAAE,CAAC;IAClB,IAAI,gBAAgB,GAAG,OAAO,CAAC;IAE/B,8CAA8C;IAC9C,OAAO,gBAAgB,CAAC,MAAM,GAAG,CAAC,EAAE;QAChC,gDAAgD;QAChD,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC;QAErD,wDAAwD;QACxD,MAAM,KAAK,GAAG,gBAAgB,CAAC,KAAK,CAAC,CAAC,EAAE,SAAS,CAAC,CAAC;QAEnD,6BAA6B;QAC7B,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAEnB,oDAAoD;QACpD,gBAAgB,GAAG,gBAAgB,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;KACxD;IAED,yDAAyD;IACzD,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE;QACxB,MAAM,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,iBAAiB;QACxE,cAAc,CAAC,KAAK,CAAC,KAAK,CAAC,CAAA;KAE9B;AACL,CAAC;AAKD,KAAK,UAAU,2BAA2B,CAAC,YAAoB,EAAE,WAAmB,EAAE,OAAe;IACjG,IAAI;QACA,MAAM,QAAQ,CAAC,IAAI,CAAC,IAAI,gCAAc,CAAC;YACnC,SAAS,EAAE,0BAA0B;YACrC,IAAI,EAAE;gBACF,YAAY,EAAE,EAAE,CAAC,EAAE,YAAY,EAAE;gBACjC,WAAW,EAAE,EAAE,CAAC,EAAE,WAAW,EAAE;gBAC/B,OAAO,EAAE,EAAE,CAAC,EAAE,OAAO,EAAE;aAC1B;SACJ,CAAC,CAAC,CAAC;QACJ,MAAM,CAAC,KAAK,CAAC,6BAA6B,CAAC,CAAC;KAC/C;IAAC,OAAO,KAAK,EAAE;QACZ,OAAO,CAAC,KAAK,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;KAClC;AACL,CAAC;AAED,KAAK,UAAU,cAAc,CAAE,KAA6B,EAAE,cAAqC;IAE/F,IAAI;QACA,MAAM,CAAC,IAAI,CAAC,KAAY,CAAC,CAAC;QAE1B,MAAM,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QACtD,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC;QACrC,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC;QAE/B,MAAM,kBAAkB,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,CAAC;QAC9F,MAAM,iBAAiB,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC;QACzF,MAAM,cAAc,GAAG,IAAI,CAAC,UAAU,IAAI,EAAE,CAAC;QAC7C,MAAM,YAAY,GAAG,IAAI,CAAC,QAAQ,IAAI,EAAE,CAAC;QAEzC,MAAM,oBAAoB,GAAG,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC3D,MAAM,mBAAmB,GAAG,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAGzD,MAAM,CAAC,WAAW,EAAE,kBAAkB,EAAE,gBAAgB,EAAE,gBAAgB,EAAE,iBAAiB,EAAE,aAAa,EAAE,iBAAiB,CAAC,GAAG,MAAM,gBAAgB,CAAC,WAAW,EAAE,QAAQ,CAAC,CAAC;QACjL,IAAI,WAAW,IAAI,kBAAkB,EAAE;YACnC,MAAM,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;SAGhC;aAAM;YACH,MAAM,CAAC,KAAK,CAAC,mCAAmC,CAAC,CAAC;YAClD,MAAM,IAAI,KAAK,CAAC,mCAAmC,CAAC,CAAC;SACxD;QAED,MAAM,SAAS,GAAG,aAAa,CAAC,WAAW,EAAE,mBAAmB,EAAE,oBAAoB,EAAE,QAAQ,CAAC,CAAC;QAElG,IAAI,cAAc,GAAG,MAAM,iBAAiB,CAAC,WAAW,EAAE,SAAS,CAAC,CAAC;QACrE,IAAI,CAAC,cAAc,EAAE;YACjB,MAAM,CAAC,IAAI,CAAC,2CAA2C,CAAC,CAAC;YACzD,MAAM,cAAc,GAAG,MAAM,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC;YACvD,MAAM,iBAAiB,GAAG,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAEpD,MAAM,UAAU,GAAG,4BAA4B,CAC3C,mBAAmB,EACnB,oBAAoB,EACpB,cAAc,EACd,YAAY,EACZ,iBAAiB,EACjB,WAAW,EACX,gBAAgB,IAAI,EAAE,EACtB,iBAAiB,IAAI,EAAE,EACvB,aAAa,IAAI,EAAE,EACnB,iBAAiB,IAAI,EAAE,EACvB,QAAS,CACZ,CAAC;YACF,cAAc,GAAG,MAAM,eAAe,CAAC,UAAU,EAAE,cAAc,CAAC,CAAC;YACnE,MAAM,2BAA2B,CAAC,WAAW,EAAE,SAAS,EAAE,cAAc,CAAC,CAAC;SAC7E;aACI;YACD,MAAM,wBAAwB,CAAC,cAAc,EAAE,cAAc,CAAC,CAAA;SAEjE;QACD,MAAM,CAAC,IAAI,CAAC,oBAAoB,cAAc,EAAE,CAAC,CAAC;KACrD;IAAC,OAAO,KAAK,EAAE;QACZ,OAAO,CAAC,KAAK,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;KAClC;IACD,cAAc,CAAC,GAAG,EAAE,CAAC;AACzB,CAAC;AAEY,QAAA,OAAO,GAAG,SAAS,CAAC,iBAAiB,CAAC,cAAc,CAAC,CAAC","sourcesContent":["import { DynamoDBClient, GetItemCommand, PutItemCommand } from \"@aws-sdk/client-dynamodb\";\nimport { unmarshall } from \"@aws-sdk/util-dynamodb\";\nimport { Tracer } from \"@aws-lambda-powertools/tracer\";\nimport { Logger } from \"@aws-lambda-powertools/logger\";\nimport { APIGatewayProxyEventV2, Handler, Context } from 'aws-lambda';\nimport { createHash } from 'crypto';\nimport { BedrockRuntimeClient, InvokeModelWithResponseStreamCommand } from \"@aws-sdk/client-bedrock-runtime\";\n\nconst logger = new Logger();\nconst dynamodb = new DynamoDBClient({});\n\nconst PRODUCT_TABLE_NAME = process.env.PRODUCT_TABLE_NAME\nconst PRODUCT_SUMMARY_TABLE_NAME = process.env.PRODUCT_SUMMARY_TABLE_NAME\nconst MODEL_ID = \"anthropic.claude-3-haiku-20240307-v1:0\"\n\n\n\nconst bedrockRuntimeClient = new BedrockRuntimeClient({ region: process.env.REGION || 'us-east-1' });\n\n\ndeclare global {\n    namespace awslambda {\n      function streamifyResponse(\n        f: (\n          event: APIGatewayProxyEventV2,\n          responseStream: NodeJS.WritableStream,\n          context: Context\n        ) => Promise<void>\n      ): Handler;\n    }\n}\n\n\n\ninterface ProductItem {\n    product_code: string;\n    language: string;\n    product_name?: string;\n    ingredients?: string;\n    additives?: string;\n    allergens_tags?: string[];\n    nutriments?: any;\n    labels_tags?: string[];\n    categories?: string;\n}\n\ninterface ProductSummaryItem {\n    product_code: string;\n    params_hash: string;\n    summary: string;\n}\n\ninterface SummaryData {\n    recommendations: string[];\n    benefits: string[];\n    disadvantages: string[];\n  }\n\n\nfunction generateProductSummaryPrompt(\n    userAllergies: string,\n    userPreference: string,\n    userHealthGoal: string,\n    userReligion: string,\n    productIngredients: string,\n    productName: string,\n    productAllergens: string[],\n    productNutriments: any,\n    productLabels: string[],\n    productCategories: string,\n    language: string\n    ): string {\n    \n    // Format nutriments for display\n    let nutrimentInfo = '';\n    if (productNutriments && Object.keys(productNutriments).length > 0) {\n        nutrimentInfo = '\\n<nutrition_per_100g>\\n';\n        if (productNutriments['energy-kcal_100g']) nutrimentInfo += `Calories: ${productNutriments['energy-kcal_100g']} kcal\\n`;\n        if (productNutriments['carbohydrates_100g']) nutrimentInfo += `Carbohydrates: ${productNutriments['carbohydrates_100g']}g\\n`;\n        if (productNutriments['sugars_100g']) nutrimentInfo += `Sugars: ${productNutriments['sugars_100g']}g\\n`;\n        if (productNutriments['fat_100g']) nutrimentInfo += `Fat: ${productNutriments['fat_100g']}g\\n`;\n        if (productNutriments['saturated-fat_100g']) nutrimentInfo += `Saturated Fat: ${productNutriments['saturated-fat_100g']}g\\n`;\n        if (productNutriments['proteins_100g']) nutrimentInfo += `Protein: ${productNutriments['proteins_100g']}g\\n`;\n        if (productNutriments['fiber_100g']) nutrimentInfo += `Fiber: ${productNutriments['fiber_100g']}g\\n`;\n        if (productNutriments['salt_100g']) nutrimentInfo += `Salt: ${productNutriments['salt_100g']}g\\n`;\n        nutrimentInfo += '</nutrition_per_100g>\\n';\n    }\n    \n    // Format allergens - only if user has allergies\n    let allergenInfo = '';\n    if (userAllergies && productAllergens && productAllergens.length > 0) {\n        allergenInfo = `\\n<product_allergens>${productAllergens.join(', ')}</product_allergens>\\n`;\n    }\n    \n    // Format labels\n    let labelInfo = '';\n    if (productLabels && productLabels.length > 0) {\n        labelInfo = `\\n<product_labels>${productLabels.join(', ')}</product_labels>\\n`;\n    }\n    \n    // Format categories\n    let categoryInfo = '';\n    if (productCategories) {\n        categoryInfo = `\\n<product_categories>${productCategories}</product_categories>\\n`;\n    }\n    \n    // Build instructions based on what user has set\n    let instructions = `You are a nutrition expert providing recommendations about a specific product.\n\n    Your task:\n    `;\n    \n    if (userAllergies) {\n        instructions += `1. CRITICAL: Check if any product allergens match the user's allergies (${userAllergies}). If there is a match, prominently warn the user.\\n`;\n    }\n    \n    if (userPreference) {\n        instructions += `${userAllergies ? '2' : '1'}. Check if product labels match dietary preferences (${userPreference}). Use labels for direct matching, or analyze categories and ingredients.\\n`;\n    }\n    \n    if (userHealthGoal) {\n        instructions += `${(userAllergies ? 1 : 0) + (userPreference ? 1 : 0) + 1}. Use nutritional data to assess if the product aligns with the health goal: ${userHealthGoal}.\\n`;\n    }\n    \n    if (userReligion) {\n        instructions += `${(userAllergies ? 1 : 0) + (userPreference ? 1 : 0) + (userHealthGoal ? 1 : 0) + 1}. Check if product labels match religious requirement: ${userReligion}.\\n`;\n    }\n    \n    instructions += `- Present three nutritional benefits and three nutritional disadvantages for the product based on actual nutritionalvalues.\n    If the user's information is not provided or is empty, offer general nutritional advice based on the product's nutritional data.\n    IMPORTANT: Only mention allergens, dietary preferences, health goals, or religious requirements if the user has specified them. Do not discuss aspects the user hasn't set.`;\n    \n    let userContext = '';\n    if (userAllergies) userContext += `\\n<user_allergies>${userAllergies}</user_allergies>`;\n    if (userHealthGoal) userContext += `\\n<user_health_goal>${userHealthGoal}</user_health_goal>`;\n    if (userPreference) userContext += `\\n<user_dietary_preferences>${userPreference}</user_dietary_preferences>`;\n    if (userReligion) userContext += `\\n<user_religious_requirement>${userReligion}</user_religious_requirement>`;\n    \n    return `Human:\n          ${instructions}\n  \n          Provide recommendation for the following product:\n            <product_name>${productName}</product_name>\n            <product_ingredients>${productIngredients}</product_ingredients>\n            <allergenInfo>${allergenInfo}</allergenInfo>\n            <labelInfo>${labelInfo}</labelInfo>\n            <categoryInfo>${categoryInfo}</categoryInfo>\n            <nutrimentInfo>${nutrimentInfo}</nutrimentInfo>\n\n          For the user:\n            ${userContext}\n          \n          Provide the response in the third person, in ${language}, skip the preambule, disregard any content at the end and provide only the response in this Markdown format:\n\n\n        markdown\n\n        Describe allergen warnings (if any), dietary label compatibility, religious requirement compatibility, health goal compatibility, dietary preference compatibility, and recommendation here combined in one single short paragraph\n\n        #### Benefits title here\n        - Describe benefits here\n\n        #### Disadvantages title here\n        - Describe disadvantages here\n          \n          Assistant:\n          `;\n}\n\nfunction generateCombinedString(obj: { [key: string]: any }): string {\n    const concatenatedString = Object.keys(obj).join('');\n    return concatenatedString;\n}\n\n\n\nfunction calculateHash(\n    productCode: string,\n    userAllergies: any,\n    userPreferenceData: any,\n    language: string\n    ): string {\n    /**\n     * Calculates a SHA-256 hash based on various input data.\n     *\n     * @param userAllergies - A string containing user allergies data.\n     * @param userPreferenceData - A string containing user preference data.\n     * @param productIngredients - A string containing product ingredients data.\n     * @param productName - The name of the product.\n     * @param language - The language.\n     * @param productAdditives - A string containing product additives data.\n     * @returns The SHA-256 hash value calculated based on the concatenated string representations of the input data.\n     */\n\n    // Convert dictionaries to JSON strings\n    const userAllergiesStr = generateCombinedString(userAllergies);//JSON.stringify(userAllergies);\n    const userPreferenceDataStr = generateCombinedString(userPreferenceData);\n    \n    // Concatenate the string representations of the variables\n    const concatenatedString = `${productCode}${userAllergiesStr}${userPreferenceDataStr}${language}`;\n    // Calculate the hash\n    const hashedValue = createHash('sha256').update(concatenatedString).digest('hex');\n    \n    return hashedValue;\n}\n\n/**\n * Retrieves product information from the database using the provided product code.\n *\n * @param productCode - The code of the product to retrieve information for.\n * @param language - The language for the product information.\n * @returns A tuple containing product name, ingredients, additives, allergens, and nutriments if the product is found in the database; otherwise, returns [null, null, null, null, null].\n */\nasync function getProductFromDb(productCode: string, language: string): Promise<[string | null, string | null, string | null, string[] | null, any | null, string[] | null, string | null]> {\n\n    try {\n        const { Item  = {} } = await dynamodb.send(new GetItemCommand({\n            TableName: PRODUCT_TABLE_NAME,\n            Key: {\n                product_code: { S: productCode },\n                language: { S: language }\n            }\n        }));\n        // Check if the item exists\n        if (Item) {\n            const item = unmarshall(Item) as ProductItem;\n            return [\n                item.product_name || null, \n                item.ingredients || null, \n                item.additives || null,\n                item.allergens_tags || null,\n                item.nutriments || null,\n                item.labels_tags || null,\n                item.categories || null\n            ];\n        } else {\n            return [null, null, null, null, null, null, null];\n        }\n    } catch (e) {\n        console.error('Error while getting the Product from database', e);\n        return [null, null, null, null, null, null, null];\n    }\n}\n\nasync function getProductSummary(productCode: string, paramsHash: string): Promise<string | null> {\n    /**\n     * Retrieves the summary of a product from the database using the product code and parameters hash.\n     *\n     * @param productCode - The code of the product.\n     * @param paramsHash - The hash value representing parameters.\n     * @returns The summary of the product if found in the database; otherwise, returns null.\n     */\n  \n    const { Item  = {} } = await dynamodb.send(new GetItemCommand({\n        TableName: PRODUCT_SUMMARY_TABLE_NAME,\n        Key: {\n            product_code: { S: productCode },\n            params_hash: { S: paramsHash }\n        }\n    }));\n  \n    if (Item) {\n      const item = unmarshall(Item) as ProductSummaryItem;\n      return item.summary;\n    } else {\n      return null;\n    }\n}\n\nasync function generateSummary(promptText: string, responseStream: NodeJS.WritableStream) {\n\n    const payload = {\n        messages: [\n            {\n                role: \"user\",\n                content: [\n                    {\n                        \"type\": \"text\",\n                        \"text\": promptText\n                    }\n                ]\n            }\n        ],\n        max_tokens: 500,\n        temperature: 0.5,\n        anthropic_version: \"bedrock-2023-05-31\"\n      };\n    const params = {\n        modelId: MODEL_ID,\n        contentType: \"application/json\",\n        accept: \"application/json\",\n        body: JSON.stringify(payload),\n    };\n    let completion = '';\n    try {\n        try {\n            const command = new InvokeModelWithResponseStreamCommand(params);\n            const response = await bedrockRuntimeClient.send(command);\n            const events = response.body;\n            for await (const event of events || []) {\n                // Check the top-level field to determine which event this is.\n                if (event.chunk) {\n                  const decoded_event = JSON.parse(\n                    new TextDecoder().decode(event.chunk.bytes),\n                  );\n                  if (decoded_event.type  === 'content_block_delta' && decoded_event.delta.type === 'text_delta'){\n                    responseStream.write(decoded_event.delta.text)\n                    completion += decoded_event.delta.text;\n                  }\n                } else {\n                  logger.error(`event = ${event}`)\n                }\n              }\n            \n              logger.info('Stream ended!')\n        } catch (err) {\n            // handle error\n            logger.error(err as any);\n        }\n    }\n    catch (e) {\n        logger.error(`Error while generating summary: ${e}`);\n        completion = \"Error while generating summary\";\n    }\n    return completion;\n}\n\nasync function simulateSummaryStreaming(content: string, responseStream: NodeJS.WritableStream): Promise<void> {\n   \n    const chunks = [];\n    let remainingContent = content;\n\n    // Loop until all content is split into chunks\n    while (remainingContent.length > 0) {\n        // Generate a random chunk size between 1 and 10\n        const chunkSize = Math.floor(Math.random() * 10) + 1;\n\n        // Take a chunk of content with the generated chunk size\n        const chunk = remainingContent.slice(0, chunkSize);\n\n        // Add the chunk to the array\n        chunks.push(chunk);\n\n        // Remove the taken chunk from the remaining content\n        remainingContent = remainingContent.slice(chunkSize);\n    }\n\n    // Simulate streaming by emitting each chunk with a delay\n    for (const chunk of chunks) {\n        await new Promise(resolve => setTimeout(resolve, 50)); // Simulate delay\n        responseStream.write(chunk)\n\n    }\n}\n\n\n\n\nasync function putProductSummaryToDynamoDB(product_code: string, params_hash: string, summary: string) {\n    try {\n        await dynamodb.send(new PutItemCommand({\n            TableName: PRODUCT_SUMMARY_TABLE_NAME,\n            Item: {\n                product_code: { S: product_code },\n                params_hash: { S: params_hash },\n                summary: { S: summary }\n            }\n        }));\n        logger.debug(\"Summary saved into database\");\n    } catch (error) {\n        console.error(\"Error:\", error);\n    }\n}\n\nasync function messageHandler (event: APIGatewayProxyEventV2, responseStream: NodeJS.WritableStream) {\n\n    try {\n        logger.info(event as any);\n\n        const body = event.body ? JSON.parse(event.body) : {};\n        const productCode = body.productCode;\n        const language = body.language;\n\n        const userPreferenceKeys = Object.keys(body.preferences).filter(key => body.preferences[key]);\n        const userAllergiesKeys = Object.keys(body.allergies).filter(key => body.allergies[key]);\n        const userHealthGoal = body.healthGoal || '';\n        const userReligion = body.religion || '';\n\n        const userPreferenceString = userPreferenceKeys.join(', ');\n        const userAllergiesString = userAllergiesKeys.join(', ');\n\n\n        const [productName, productIngredients, productAdditives, productAllergens, productNutriments, productLabels, productCategories] = await getProductFromDb(productCode, language);\n        if (productName && productIngredients) {\n            logger.info(\"Product found\");\n\n\n        } else {\n            logger.error(\"Product not found in the database\");\n            throw new Error('Product not found in the database');\n        }\n\n        const hashValue = calculateHash(productCode, userAllergiesString, userPreferenceString, language);\n\n        let productSummary = await getProductSummary(productCode, hashValue);\n        if (!productSummary) {        \n            logger.info(\"Product Summary not found in the database\");\n            const ingredientKeys = Object.keys(productIngredients);\n            const ingredientsString = ingredientKeys.join(', ');\n\n            const promptText = generateProductSummaryPrompt(\n                userAllergiesString,\n                userPreferenceString,\n                userHealthGoal,\n                userReligion,\n                ingredientsString,\n                productName,\n                productAllergens || [],\n                productNutriments || {},\n                productLabels || [],\n                productCategories || '',\n                language!\n            );\n            productSummary = await generateSummary(promptText, responseStream);\n            await putProductSummaryToDynamoDB(productCode, hashValue, productSummary);\n        }\n        else {\n            await simulateSummaryStreaming(productSummary, responseStream)\n\n        }\n        logger.info(`Product Summary: ${productSummary}`);\n    } catch (error) {\n        console.error(\"Error:\", error);\n    }\n    responseStream.end();\n}\n\nexport const handler = awslambda.streamifyResponse(messageHandler);"]} \ No newline at end of file diff --git a/lambda/recipe_step_by_step/index.js b/lambda/recipe_step_by_step/index.js index 90a1f04..b45e518 100644 --- a/lambda/recipe_step_by_step/index.js +++ b/lambda/recipe_step_by_step/index.js @@ -134,4 +134,4 @@ async function messageHandler(event, responseStream) { responseStream.end(); } exports.handler = awslambda.streamifyResponse(messageHandler); -//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import { Tracer } from \"@aws-lambda-powertools/tracer\";\nimport { Logger } from \"@aws-lambda-powertools/logger\";\nimport { APIGatewayProxyEventV2, Handler, Context } from 'aws-lambda';\nimport { createHash } from 'crypto';\nimport { BedrockRuntimeClient, InvokeModelWithResponseStreamCommand } from \"@aws-sdk/client-bedrock-runtime\";\n\ndeclare global {\n    namespace awslambda {\n      function streamifyResponse(\n        f: (\n          event: APIGatewayProxyEventV2,\n          responseStream: NodeJS.WritableStream,\n          context: Context\n        ) => Promise<void>\n      ): Handler;\n    }\n}\n\n\nconst MODEL_ID = \"anthropic.claude-3-haiku-20240307-v1:0\"\n\nconst tracer = new Tracer();\nconst logger = new Logger();\n\n\n\nconst bedrockRuntimeClient = new BedrockRuntimeClient({ region: process.env.REGION || 'us-east-1' });\n\n\n\n\nasync function generateRecipeSteps(language: string, recipe, responseStream) {\n\n    const systemPrompt = \"Your task is to generate personalized recipe ideas based on the user's input of available ingredients and dietary preferences. Use this information to suggest a variety of creative and delicious recipes that can be made using the given ingredients while accommodating the user's dietary needs, if any are mentioned. For each recipe, provide a brief description, a list of required ingredients, and a simple set of instructions. Ensure that the recipes are easy to follow, nutritious, and can be prepared with minimal additional ingredients or equipment.\";\n\n\n    const promptText = `\n    Recipee title:${recipe.title}\n    Recipee description:${recipe.description}\n    Available ingredients:${recipe.ingredients} ${recipe.optional_ingredients}\n    \n    Answer must be in the following markdown format:\n    ### Step 1: [Step Title]\n    - Action 1: [Action description] \n    - Action 2: [Action description]\n\n    **Ingredients:** [Ingredient 1], [Ingredient 2], [Ingredient 3]\n\n    ### Step 2: [Step Title]\n    - Action 1: [Action description]\n    - Action 2: [Action description]\n\n    **Ingredients:** [Ingredient 1], [Ingredient 2]\n\n    ### Step 3: [Step Title]\n    - Action 1: [Action description]\n    - Action 2: [Action description]\n\n    **Ingredients:** [Ingredient 1], [Ingredient 2], [Ingredient 3], [Ingredient 4]\n\n    Describe the actions in each step with detailed but concise descriptions, including ingredients needed, quantities, time, and any appliances required. Ensure your tone is engaging and friendly.\n    \n    Only use ingredients present in the provided recipe.\n\n    Response must be in ${language}.\n\n    Think step by step and elaborate your thoughts inside <thinking></thinking> then answer in a markdown format`;\n\n    const payload = {\n        messages: [\n            {\n                role: \"user\",\n                content: [\n                    {\n                        \"type\": \"text\",\n                        \"text\": promptText\n                    }\n                ]\n            }\n        ],\n        max_tokens: 1000,\n        system: systemPrompt,\n        temperature: 0.5,\n        stop_sequences: ['</answer>'],\n        anthropic_version: \"bedrock-2023-05-31\"\n      };\n    const params = {\n        modelId: MODEL_ID,\n        contentType: \"application/json\",\n        accept: \"application/json\",\n        body: JSON.stringify(payload),\n    };\n    let completion = '';\n    try {\n        try {\n            let accumulating = true;\n            let accumulatedChunks = '';\n            const command = new InvokeModelWithResponseStreamCommand(params);\n            const response = await bedrockRuntimeClient.send(command);\n            const events = response.body;\n            for await (const event of events || []) {\n                // Check the top-level field to determine which event this is.\n                if (event.chunk) {\n                  const decoded_event = JSON.parse(\n                    new TextDecoder().decode(event.chunk.bytes),\n                  );\n                  if (decoded_event.type  === 'content_block_delta' && decoded_event.delta.type === 'text_delta'){\n                    \n                    const text = decoded_event.delta.text;\n                    console.log(\"text=\"+text)\n\n                    //responseStream.write(decoded_event.delta.text)\n                    //accumulatedChunks += text;\n                    logger.info(decoded_event.delta.text);\n                    completion += decoded_event.delta.text;\n\n                    if(accumulating){\n                        accumulatedChunks += text;\n                        console.log(\"accumulatedChunks=\"+accumulatedChunks)\n                        if (accumulatedChunks.includes('</thinking>')) {\n                            console.log(\"tag found\")\n                            accumulating = false;\n                            const startIndex = accumulatedChunks.indexOf(\"</thinking>\") + \"</thinking>\".length;\n                            const remainingText = accumulatedChunks.substring(startIndex);\n                            logger.info(remainingText);\n                            responseStream.write(remainingText);\n                          }\n\n                      }else{\n                        console.log(\"responseStream write text=\"+text)\n                        responseStream.write(text)\n                      }\n\n                  }\n\n\n                } else {\n                  logger.error(`event = ${event}`)\n                }\n              }\n            \n            \n              logger.info('Stream ended!')\n        } catch (err) {\n            // handle error\n            logger.error(err as any);\n        }\n    }\n    catch (e) {\n        logger.error(`Error while generating summary: ${e}`);\n        completion = \"Error while generating summary\";\n    }\n    return completion;\n}\n\n\n\nasync function messageHandler (event, responseStream) {\n\n    try {\n        logger.info(event as any);\n\n        const body = event.body ? JSON.parse(event.body) : {};\n        const language = body.language;\n        const recipe = body.recipe;\n        const recipeSteps = await generateRecipeSteps(language, recipe, responseStream);\n        //await putProductSummaryToDynamoDB(productCode, hashValue, productSummary);\n        \n    } catch (error) {\n        console.error(\"Error:\", error);\n    }\n    responseStream.end();\n}\n\nexport const handler = awslambda.streamifyResponse(messageHandler);"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import { Tracer } from \"@aws-lambda-powertools/tracer\";\nimport { Logger } from \"@aws-lambda-powertools/logger\";\nimport { APIGatewayProxyEventV2, Handler, Context } from 'aws-lambda';\nimport { createHash } from 'crypto';\nimport { BedrockRuntimeClient, InvokeModelWithResponseStreamCommand } from \"@aws-sdk/client-bedrock-runtime\";\n\ndeclare global {\n    namespace awslambda {\n      function streamifyResponse(\n        f: (\n          event: APIGatewayProxyEventV2,\n          responseStream: NodeJS.WritableStream,\n          context: Context\n        ) => Promise<void>\n      ): Handler;\n    }\n}\n\n\nconst MODEL_ID = \"anthropic.claude-3-haiku-20240307-v1:0\"\n\nconst tracer = new Tracer();\nconst logger = new Logger();\n\n\n\nconst bedrockRuntimeClient = new BedrockRuntimeClient({ region: process.env.REGION || 'us-east-1' });\n\n\n\n\nasync function generateRecipeSteps(language: string, recipe: any, responseStream: NodeJS.WritableStream) {\n\n    const systemPrompt = \"Your task is to generate personalized recipe ideas based on the user's input of available ingredients and dietary preferences. Use this information to suggest a variety of creative and delicious recipes that can be made using the given ingredients while accommodating the user's dietary needs, if any are mentioned. For each recipe, provide a brief description, a list of required ingredients, and a simple set of instructions. Ensure that the recipes are easy to follow, nutritious, and can be prepared with minimal additional ingredients or equipment.\";\n\n\n    const promptText = `\n    Recipee title:${recipe.title}\n    Recipee description:${recipe.description}\n    Available ingredients:${recipe.ingredients} ${recipe.optional_ingredients}\n    \n    Answer must be in the following markdown format:\n    ### Step 1: [Step Title]\n    - Action 1: [Action description] \n    - Action 2: [Action description]\n\n    **Ingredients:** [Ingredient 1], [Ingredient 2], [Ingredient 3]\n\n    ### Step 2: [Step Title]\n    - Action 1: [Action description]\n    - Action 2: [Action description]\n\n    **Ingredients:** [Ingredient 1], [Ingredient 2]\n\n    ### Step 3: [Step Title]\n    - Action 1: [Action description]\n    - Action 2: [Action description]\n\n    **Ingredients:** [Ingredient 1], [Ingredient 2], [Ingredient 3], [Ingredient 4]\n\n    Describe the actions in each step with detailed but concise descriptions, including ingredients needed, quantities, time, and any appliances required. Ensure your tone is engaging and friendly.\n    \n    Only use ingredients present in the provided recipe.\n\n    Response must be in ${language}.\n\n    Think step by step and elaborate your thoughts inside <thinking></thinking> then answer in a markdown format`;\n\n    const payload = {\n        messages: [\n            {\n                role: \"user\",\n                content: [\n                    {\n                        \"type\": \"text\",\n                        \"text\": promptText\n                    }\n                ]\n            }\n        ],\n        max_tokens: 1000,\n        system: systemPrompt,\n        temperature: 0.5,\n        stop_sequences: ['</answer>'],\n        anthropic_version: \"bedrock-2023-05-31\"\n      };\n    const params = {\n        modelId: MODEL_ID,\n        contentType: \"application/json\",\n        accept: \"application/json\",\n        body: JSON.stringify(payload),\n    };\n    let completion = '';\n    try {\n        try {\n            let accumulating = true;\n            let accumulatedChunks = '';\n            const command = new InvokeModelWithResponseStreamCommand(params);\n            const response = await bedrockRuntimeClient.send(command);\n            const events = response.body;\n            for await (const event of events || []) {\n                // Check the top-level field to determine which event this is.\n                if (event.chunk) {\n                  const decoded_event = JSON.parse(\n                    new TextDecoder().decode(event.chunk.bytes),\n                  );\n                  if (decoded_event.type  === 'content_block_delta' && decoded_event.delta.type === 'text_delta'){\n                    \n                    const text = decoded_event.delta.text;\n                    console.log(\"text=\"+text)\n\n                    //responseStream.write(decoded_event.delta.text)\n                    //accumulatedChunks += text;\n                    logger.info(decoded_event.delta.text);\n                    completion += decoded_event.delta.text;\n\n                    if(accumulating){\n                        accumulatedChunks += text;\n                        console.log(\"accumulatedChunks=\"+accumulatedChunks)\n                        if (accumulatedChunks.includes('</thinking>')) {\n                            console.log(\"tag found\")\n                            accumulating = false;\n                            const startIndex = accumulatedChunks.indexOf(\"</thinking>\") + \"</thinking>\".length;\n                            const remainingText = accumulatedChunks.substring(startIndex);\n                            logger.info(remainingText);\n                            responseStream.write(remainingText);\n                          }\n\n                      }else{\n                        console.log(\"responseStream write text=\"+text)\n                        responseStream.write(text)\n                      }\n\n                  }\n\n\n                } else {\n                  logger.error(`event = ${event}`)\n                }\n              }\n            \n            \n              logger.info('Stream ended!')\n        } catch (err) {\n            // handle error\n            logger.error(err as any);\n        }\n    }\n    catch (e) {\n        logger.error(`Error while generating summary: ${e}`);\n        completion = \"Error while generating summary\";\n    }\n    return completion;\n}\n\n\n\nasync function messageHandler (event: APIGatewayProxyEventV2, responseStream: NodeJS.WritableStream) {\n\n    try {\n        logger.info(event as any);\n\n        const body = event.body ? JSON.parse(event.body) : {};\n        const language = body.language;\n        const recipe = body.recipe;\n        const recipeSteps = await generateRecipeSteps(language, recipe, responseStream);\n        //await putProductSummaryToDynamoDB(productCode, hashValue, productSummary);\n        \n    } catch (error) {\n        console.error(\"Error:\", error);\n    }\n    responseStream.end();\n}\n\nexport const handler = awslambda.streamifyResponse(messageHandler);"]} \ No newline at end of file diff --git a/resources/ui/src/assets/i18n/all.ts b/resources/ui/src/assets/i18n/all.ts index 3796a08..fc9e64d 100644 --- a/resources/ui/src/assets/i18n/all.ts +++ b/resources/ui/src/assets/i18n/all.ts @@ -38,6 +38,8 @@ const customTranslations: Record> = { ingredients_desc_additive: "Click on each additive to view AI-generated detailed descriptions", ingredients_no_additive: "The product does not have additives", + allergen_warning_title: "Allergen Warning", + allergen_warning_message: "This product contains allergens you're sensitive to:", summary_title: "Summary of ingredients generated by AI", summary_benefits_title: "Benefits", summary_disadvantages_title: "Disadvantages", @@ -128,6 +130,8 @@ const customTranslations: Record> = { ingredients_desc_additive: "Cliquez sur chaque additif pour voir les descriptions détaillées générées par l'IA", ingredients_no_additive: "Le produit ne contient pas d'additifs", + allergen_warning_title: "Avertissement Allergène", + allergen_warning_message: "Ce produit contient des allergènes auxquels vous êtes sensible :", summary_title: "Résumé des ingrédients généré par IA", summary_benefits_title: "Avantages", summary_disadvantages_title: "Inconvénients", @@ -220,6 +224,8 @@ const customTranslations: Record> = { ingredients_no_additive: "Il prodotto non contiene additivi", ingredients_desc_additive: "Fai clic su ogni additivo per visualizzare le descrizioni dettagliate generate dall’AI", + allergen_warning_title: "Avviso Allergeni", + allergen_warning_message: "Questo prodotto contiene allergeni a cui sei sensibile:", summary_title: "Riepilogo degli ingredienti generato dall’AI", summary_benefits_title: "Benefici", summary_disadvantages_title: "Svantaggi", @@ -313,6 +319,8 @@ const customTranslations: Record> = { ingredients_no_additive: "El producto no contiene aditivos", ingredients_desc_additive: "Haz clic en cada aditivo para ver descripciones detalladas generadas por IA", + allergen_warning_title: "Advertencia de Alérgenos", + allergen_warning_message: "Este producto contiene alérgenos a los que eres sensible:", summary_title: "Resumen de ingredientes generados por IA", summary_benefits_title: "Beneficios", summary_disadvantages_title: "Desventajas", @@ -398,6 +406,8 @@ arabic:{ "ingredients_desc_ingredient": "انقر على كل مكون لعرض الوصف التفصيلي الناتج عن الذكاء الاصطناعي", "ingredients_desc_additive": "انقر على كل إضافة لعرض الوصف التفصيلي الناتج عن الذكاء الاصطناعي", "ingredients_no_additive": "المنتج لا يحتوي على إضافات", + "allergen_warning_title": "تحذير من مسببات الحساسية", + "allergen_warning_message": "يحتوي هذا المنتج على مسببات حساسية تؤثر عليك:", "summary_title": "ملخص المكونات الناتجة عن الذكاء الاصطناعي", "summary_benefits_title": "الفوائد", "summary_disadvantages_title": "العيوب", diff --git a/resources/ui/src/pages/components/barcode_ingredients.tsx b/resources/ui/src/pages/components/barcode_ingredients.tsx index e9f1038..ac69854 100644 --- a/resources/ui/src/pages/components/barcode_ingredients.tsx +++ b/resources/ui/src/pages/components/barcode_ingredients.tsx @@ -39,9 +39,7 @@ const BarcodeIngredients: React.FC = ({ const [ingredientsError, setIngredientsError] = useState(""); const fetchData = async () => { - console.log( - `call backend with scannedCode: ${productCode} and language: ${language}` - ); + setApiResponse(null); setLoading(true); @@ -53,13 +51,11 @@ const BarcodeIngredients: React.FC = ({ ); if (!response.error) { - console.log("response=" + JSON.stringify(response)); const keyValueArray = Object.entries(response.ingredients_description); const newIngredients = keyValueArray.map(([key, value]) => ({ label: key, description: value, })); - console.log(newIngredients); setIngredients(newIngredients); diff --git a/resources/ui/src/pages/components/barcode_product_summary.tsx b/resources/ui/src/pages/components/barcode_product_summary.tsx index b2f013c..f1e520a 100644 --- a/resources/ui/src/pages/components/barcode_product_summary.tsx +++ b/resources/ui/src/pages/components/barcode_product_summary.tsx @@ -142,12 +142,14 @@ const BarcodeProductSummary: React.FC = ({ return (
{loadingSummary && ( -
+
- Loading Summary of Ingredients... +
+ {currentTranslations["summary_title"].replace("Summary of ingredients generated by AI", "Loading AI Analysis")}... +
- )}{" "} + )} {summaryError ? ( {summaryError} @@ -156,42 +158,64 @@ const BarcodeProductSummary: React.FC = ({ <> {recommendation && (
- - - - + + {/* AI Summary Card */} +
+
+ +

{currentTranslations["summary_title"]} - - } - - > - - - - {currentTranslations['image_title']} +

+
+
+ +
+
+ + {/* AI Generated Image Card */} + {(image || loadingImage) && ( +
+
+
+ 🎨 +

+ {currentTranslations['image_title']} +

- ), - height: "100%", - }} - > - {currentTranslations['image_title']} - - - +
+
+ {currentTranslations['image_title']} +
+
+ )}
)} From 4751d501ebe6fb2ea4679cc0edb2cc587f184094 Mon Sep 17 00:00:00 2001 From: Jeremy Labrado Date: Mon, 8 Dec 2025 09:07:23 +0100 Subject: [PATCH 2/2] feat: modernize barcode scanning UI/UX with tabs, nutritional cards, and allergen detection - Add tab-based navigation (AI Summary, Ingredients, Additives) with translations in 5 languages - Display compact nutritional info cards (calories, salt, sugar, protein) in horizontal row for mobile - Implement visual allergen flagging with red gradient borders and warning icons - Add multi-language allergen keyword detection (English, French, Spanish, Italian, Arabic) - Enhance empty state with phone icon, clear instructions, and numbered step badges - Add preference check with warning alert before scanning - Improve loading states with split feedback (scan confirmation + spinner) - Add prominent allergen warning banner with translated messages - Replace side-by-side layout with mobile-friendly tabs - Add search icon to scan button --- resources/ui/src/assets/i18n/all.ts | 15 + resources/ui/src/pages/components/barcode.tsx | 131 ++++++-- .../pages/components/barcode_ingredients.tsx | 300 ++++++++++++------ 3 files changed, 337 insertions(+), 109 deletions(-) diff --git a/resources/ui/src/assets/i18n/all.ts b/resources/ui/src/assets/i18n/all.ts index fc9e64d..9d9c2d8 100644 --- a/resources/ui/src/assets/i18n/all.ts +++ b/resources/ui/src/assets/i18n/all.ts @@ -38,6 +38,9 @@ const customTranslations: Record> = { ingredients_desc_additive: "Click on each additive to view AI-generated detailed descriptions", ingredients_no_additive: "The product does not have additives", + tab_ai_summary: "AI Summary", + tab_ingredients: "Ingredients", + tab_additives: "Additives", allergen_warning_title: "Allergen Warning", allergen_warning_message: "This product contains allergens you're sensitive to:", summary_title: "Summary of ingredients generated by AI", @@ -130,6 +133,9 @@ const customTranslations: Record> = { ingredients_desc_additive: "Cliquez sur chaque additif pour voir les descriptions détaillées générées par l'IA", ingredients_no_additive: "Le produit ne contient pas d'additifs", + tab_ai_summary: "Résumé IA", + tab_ingredients: "Ingrédients", + tab_additives: "Additifs", allergen_warning_title: "Avertissement Allergène", allergen_warning_message: "Ce produit contient des allergènes auxquels vous êtes sensible :", summary_title: "Résumé des ingrédients généré par IA", @@ -222,6 +228,9 @@ const customTranslations: Record> = { ingredients_desc_ingredient: "Fai clic su ogni ingrediente per visualizzare le descrizioni dettagliate generate dall’AI", ingredients_no_additive: "Il prodotto non contiene additivi", + tab_ai_summary: "Riepilogo IA", + tab_ingredients: "Ingredienti", + tab_additives: "Additivi", ingredients_desc_additive: "Fai clic su ogni additivo per visualizzare le descrizioni dettagliate generate dall’AI", allergen_warning_title: "Avviso Allergeni", @@ -317,6 +326,9 @@ const customTranslations: Record> = { ingredients_desc_ingredient: "Haz clic en cada ingrediente para ver descripciones detalladas generadas por IA", ingredients_no_additive: "El producto no contiene aditivos", + tab_ai_summary: "Resumen IA", + tab_ingredients: "Ingredientes", + tab_additives: "Aditivos", ingredients_desc_additive: "Haz clic en cada aditivo para ver descripciones detalladas generadas por IA", allergen_warning_title: "Advertencia de Alérgenos", @@ -406,6 +418,9 @@ arabic:{ "ingredients_desc_ingredient": "انقر على كل مكون لعرض الوصف التفصيلي الناتج عن الذكاء الاصطناعي", "ingredients_desc_additive": "انقر على كل إضافة لعرض الوصف التفصيلي الناتج عن الذكاء الاصطناعي", "ingredients_no_additive": "المنتج لا يحتوي على إضافات", + "tab_ai_summary": "ملخص الذكاء الاصطناعي", + "tab_ingredients": "المكونات", + "tab_additives": "الإضافات", "allergen_warning_title": "تحذير من مسببات الحساسية", "allergen_warning_message": "يحتوي هذا المنتج على مسببات حساسية تؤثر عليك:", "summary_title": "ملخص المكونات الناتجة عن الذكاء الاصطناعي", diff --git a/resources/ui/src/pages/components/barcode.tsx b/resources/ui/src/pages/components/barcode.tsx index 576d81b..98812a3 100644 --- a/resources/ui/src/pages/components/barcode.tsx +++ b/resources/ui/src/pages/components/barcode.tsx @@ -4,6 +4,7 @@ import Button from "@cloudscape-design/components/button"; import Ingredients from "./barcode_ingredients"; import Badge from "@cloudscape-design/components/badge"; import Link from "@cloudscape-design/components/link"; +import Alert from "@cloudscape-design/components/alert"; import { Box, Container, @@ -43,10 +44,17 @@ const Barcode: React.FC = () => { const [productCode, setProductCode] = useState(""); const [showScanner, setShowScanner] = useState(false); const [tempProductCode, setTempProductCode] = useState(""); + const [hasPreferences, setHasPreferences] = useState(false); const currentTranslations = customTranslations[language]; // Get translations for the current language or fallback to English let html5QrcodeScanner: any; + // Check if user has set preferences + useEffect(() => { + const stored = localStorage.getItem("userPreferences"); + setHasPreferences(!!stored); + }, []); + function onScanFailure(error: unknown) { console.warn(`Code scan error = ${error}`); } @@ -122,7 +130,11 @@ const Barcode: React.FC = () => { >
{!showScanner && ( - )} @@ -153,27 +165,106 @@ const Barcode: React.FC = () => {
{!showScanner && ( - -
-

{currentTranslations["scan_main_title"]}

- - -
-

- 1{" "} - {currentTranslations["scan_label_1"]}{" "} - - {currentTranslations["scan_label_2"]} - -

-

- 2{" "} - {currentTranslations["scan_label_3"]} -

+
+ {!hasPreferences && ( + + To get personalized nutritional information, please{" "} + set your preferences before scanning. + + )} + +
+
+ 📱 +
+

+ {currentTranslations["scan_main_title"]} +

+

+ Point your camera at a product barcode +

+
+ +
+
+
+ 1 +
+ + {currentTranslations["scan_label_1"]}{" "} + + {currentTranslations["scan_label_2"]} + + +
+ +
+
+ 2
- + + {currentTranslations["scan_label_3"]} + +
- +
)}
diff --git a/resources/ui/src/pages/components/barcode_ingredients.tsx b/resources/ui/src/pages/components/barcode_ingredients.tsx index ac69854..72854f0 100644 --- a/resources/ui/src/pages/components/barcode_ingredients.tsx +++ b/resources/ui/src/pages/components/barcode_ingredients.tsx @@ -6,7 +6,7 @@ import Button from "@cloudscape-design/components/button"; import TextContent from "@cloudscape-design/components/text-content"; import Spinner from "@cloudscape-design/components/spinner"; import Alert from "@cloudscape-design/components/alert"; -import { ColumnLayout, Container } from "@cloudscape-design/components"; +import { Container, Tabs, Box, ColumnLayout } from "@cloudscape-design/components"; import Header from "@cloudscape-design/components/header"; import { SpaceBetween } from "@cloudscape-design/components"; import { callAPI } from "../../assets/js/custom"; @@ -37,6 +37,43 @@ const BarcodeIngredients: React.FC = ({ const [loading, setLoading] = useState(true); // Added loading state const [productName, setProductName] = useState(true); // Added loading state const [ingredientsError, setIngredientsError] = useState(""); + const [nutriments, setNutriments] = useState(null); + const [allergensTags, setAllergensTags] = useState([]); + + // Check if ingredient matches user allergens + const isAllergen = (ingredientLabel: string): boolean => { + const stored = localStorage.getItem("userPreferences"); + if (!stored) return false; + + try { + const prefs = JSON.parse(stored); + const userAllergies = prefs.allergies || []; + + // Check if ingredient label contains any user allergen + const lowerLabel = ingredientLabel.toLowerCase(); + + // Common allergen keywords in multiple languages + const allergenKeywords: Record = { + milk: ["milk", "lait", "leche", "latte"], + eggs: ["egg", "oeuf", "huevo", "uovo"], + peanuts: ["peanut", "arachide", "cacahuete", "arachidi"], + tree_nuts: ["nut", "noix", "nuez", "noci", "almond", "amande", "cashew", "cajou"], + soy: ["soy", "soja", "soia"], + wheat: ["wheat", "blé", "trigo", "grano"], + fish: ["fish", "poisson", "pescado", "pesce"], + shellfish: ["shellfish", "crustacé", "marisco", "crostacei", "shrimp", "crevette"], + sesame: ["sesame", "sésame", "sésamo", "sesamo"] + }; + + return userAllergies.some((allergy: any) => { + const allergyValue = allergy.value.toLowerCase(); + const keywords = allergenKeywords[allergyValue] || [allergyValue]; + return keywords.some(keyword => lowerLabel.includes(keyword)); + }); + } catch { + return false; + } + }; const fetchData = async () => { @@ -60,6 +97,8 @@ const BarcodeIngredients: React.FC = ({ setIngredients(newIngredients); setProductName(response.product_name); + setNutriments(response.nutriments || null); + setAllergensTags(response.allergens_tags || []); const myAdditives:Additive [] = []; for (const key in response.additives_description) { @@ -98,11 +137,19 @@ const BarcodeIngredients: React.FC = ({ {loading && (
- + - {currentTranslations["scan_scanned_label"]}:{" "} - {productCode} | - {currentTranslations["scan_scanning_label"]}... +
+ + {currentTranslations["scan_scanned_label"]}:{" "} + {productCode} +
+
+ +
+ + {currentTranslations["scan_scanning_label"]}... +
@@ -118,94 +165,169 @@ const BarcodeIngredients: React.FC = ({ {apiResponse && (
- -
- - - {currentTranslations["scan_scanned_label"]}:{" "} - {productCode} |{" "} - {currentTranslations["product_name_label"]}:{" "} - {productName} - - -
- - ({ - id: `${item.id}`, - content: ( - - - - ), - }))} - /> - } - header={ -
- {currentTranslations["ingredients_title1"]} -
- } - > -

- {" "} - {currentTranslations["ingredients_desc_ingredient"]} + {/* Product Header Card */} + + +

+

+ {productName} +

+

+ {currentTranslations["scan_scanned_label"]}: {productCode}

- +
- {additives && ( - 0 ? ( - ({ - id: `${item.id}`, - content: ( - - - - ), - }))} - /> - ) : null - } - header={ -
- {currentTranslations["ingredients_title2"]} -
- } - > - {additives.length > 0 ? ( -

- {currentTranslations["ingredients_desc_additive"]} -

- ) : ( -

The product does not have additives

- )} - -
+ {/* Nutritional Info Cards */} + {nutriments && ( +
+
+
🔥
+
+ {nutriments["energy-kcal_100g"] || "N/A"} +
+
kcal
+
+
+
🧂
+
+ {nutriments["salt_100g"] ? `${nutriments["salt_100g"]}g` : "N/A"} +
+
Salt
+
+
+
🍬
+
+ {nutriments["sugars_100g"] ? `${nutriments["sugars_100g"]}g` : "N/A"} +
+
Sugar
+
+
+
💪
+
+ {nutriments["proteins_100g"] ? `${nutriments["proteins_100g"]}g` : "N/A"} +
+
Protein
+
+
+ )} + + {/* Allergen Warning */} + {allergensTags && allergensTags.length > 0 && ( + + ⚠️ {currentTranslations["allergen_warning_title"]} +
+ {currentTranslations["allergen_warning_message"]} {allergensTags.map(tag => tag.replace("en:", "")).join(", ")} +
)} -
-
+
+ - + {/* Tabs for Ingredients, Additives, and AI Summary */} + + ), + }, + { + label: currentTranslations["tab_ingredients"], + id: "ingredients", + content: ( + + +

+ {currentTranslations["ingredients_desc_ingredient"]} +

+ { + const isAllergenItem = isAllergen(item.label); + return { + id: `${item.id}`, + content: ( + +
+ +
+
+ ), + }; + })} + /> +
+
+ ), + }, + { + label: currentTranslations["tab_additives"], + id: "additives", + content: ( + + + {additives.length > 0 ? ( + <> +

+ {currentTranslations["ingredients_desc_additive"]} +

+ ({ + id: `${item.id}`, + content: ( + + + + ), + }))} + /> + + ) : ( +

The product does not have additives

+ )} +
+
+ ), + }, + ]} + />
)}