From 4af35674cc2cd9f13fd14fe2037ffd3026d187b8 Mon Sep 17 00:00:00 2001 From: Korey Stegared-Pace Date: Wed, 21 Feb 2024 22:28:51 +0100 Subject: [PATCH] Added new video links and 2 new lessons --- .../translations/cn/README.md | 56 +-- .../translations/es-mx/README.md | 57 +-- .../translations/ja-jp/README.md | 101 ++-- .../translations/pt-br/README.md | 64 +-- .../translations/cn/README.md | 14 +- .../translations/ja-jp/README.md | 74 +-- .../translations/pt-br/README.md | 5 +- 04-prompt-engineering-fundamentals/README.md | 4 +- .../translations/cn/README.md | 107 ++-- .../translations/ja-jp/README.md | 174 +++---- .../translations/pt-br/README.md | 74 ++- 05-advanced-prompts/README.md | 160 +++--- 05-advanced-prompts/translations/cn/README.md | 166 +++---- .../translations/ja-jp/README.md | 177 ++++--- .../translations/pt-br/README.md | 146 +++--- 06-text-generation-apps/README.md | 453 +++++++++-------- .../translations/cn/README.md | 460 +++++++++--------- .../translations/ja-jp/README.md | 75 ++- .../translations/pt-br/README.md | 406 ++++++++-------- 07-building-chat-applications/README.md | 18 +- .../translations/cn/README.md | 73 ++- .../translations/ja-jp/README.md | 102 ++-- .../translations/pt-br/README.md | 63 +-- .../README.md | 28 +- .../translations/cn/README.md | 56 +-- .../translations/ja-jp/README.md | 28 +- .../translations/pt-br/README.md | 26 +- 12-designing-ux-for-ai-applications/README.md | 26 +- .../translations/cn/README.md | 28 +- .../translations/ja-jp/README.md | 26 +- .../translations/pt-br/README.md | 16 +- 13-securing-ai-applications/README.md | 164 +++++++ .../images/13-AI-red-team.png | Bin 0 -> 173763 bytes .../images/13-lesson-banner.jpg | Bin 0 -> 230536 bytes 17-ai-agents/README.md | 148 ++++++ README.md | 101 ++-- 36 files changed, 1980 insertions(+), 1696 deletions(-) create mode 100644 13-securing-ai-applications/README.md create mode 100644 13-securing-ai-applications/images/13-AI-red-team.png create mode 100644 13-securing-ai-applications/images/13-lesson-banner.jpg create mode 100644 17-ai-agents/README.md diff --git a/01-introduction-to-genai/translations/cn/README.md b/01-introduction-to-genai/translations/cn/README.md index eda2b5191..fd8ed1f6b 100644 --- a/01-introduction-to-genai/translations/cn/README.md +++ b/01-introduction-to-genai/translations/cn/README.md @@ -1,8 +1,8 @@ # 第一章 : 生成式人工智能和 LLMs 介绍 -[![Introduction to Generative AI and Large Language Models](../../images/01-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/vf_mZrn8ibc?WT.mc_id=academic-105485-koreyst) +[![Introduction to Generative AI and Large Language Models](../../images/01-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=36c6795a-e63c-46dd-8d69-df8bbe6e7bc9?WT.mc_id=academic-105485-koreyst) -*(点击该图片看本章导学视频)* +_(点击该图片看本章导学视频)_ 生成式人工智能是能够生成文本、图像和其他类型内容的人工智能。 它之所以成为一项出色的技术,是因为它使人工智能更通用化,任何人都可以使用它,只需简单文本提示,使用自然语言编写的提示词。 你不需要学习 Java 或 SQL 这样的语言来完成一些有价值的事情,你只需要使用你的日常语言,描述需要的东西,然后 LLMs 就会给出相关建议。 其应用场景和影响是巨大的,您可以在几秒钟内完成写作或理解报告、编写应用程序等。 @@ -12,23 +12,23 @@ 本课程将包含以下内容: -* 业务场景介绍:我们的创业理念和使命。 -* 生成式人工智能以及我们如何适应当前的技术格局。 -* LLMs 的内部工作原理。 -* LLMs 的主要功能和实际用例。 +- 业务场景介绍:我们的创业理念和使命。 +- 生成式人工智能以及我们如何适应当前的技术格局。 +- LLMs 的内部工作原理。 +- LLMs 的主要功能和实际用例。 ## 学习目标 学习完本课程后,您将会了解到: -* 什么是生成式人工智能以及 LLMs 如何工作。 -* 如何针对不同的使用 LLMs 的例子(这里我们重点关注教育场景)。 +- 什么是生成式人工智能以及 LLMs 如何工作。 +- 如何针对不同的使用 LLMs 的例子(这里我们重点关注教育场景)。 -## 场景: our educational startup +## 场景: our educational startup 生成人工智能 (AI) 代表了人工智能技术的巅峰,突破了曾经被认为不可能的界限。 生成式人工智能模型具有多模态的功能,在本课程中,我们将会讲述一间虚构的初创公司利用生成式人工智能改变传统教育方式的故事。 我们将这家初创公司称为“Our startup”。 Our startup 致力于教育领域,其雄心勃勃的使命宣言是 -> *在全球范围内提高学习的可及性,确保公平的受教育机会,并根据每个学习者的需求为他们提供个性化的学习体验*。 +> _在全球范围内提高学习的可及性,确保公平的受教育机会,并根据每个学习者的需求为他们提供个性化的学习体验_。 我们的初创团队意识到,如果不利用现代最强大的工具之—— LLM,我们将无法实现这一目标。 @@ -63,7 +63,7 @@ ![AI、ML、DL 和生成式 AI](../../images/AI-diagram.png?WT.mc_id=academic-105485-koreyst) -经过人工智能领域数十年的研究,一种名为 *Transformer* 的新模型架构克服了 RNN 的限制,能够获得更长的文本序列作为输入。 Transformer 基于注意力机制,使模型能够为其接收到的输入赋予不同的权重,“更加专注于”关联信息集中的地方,不管它们在文本序列中的顺序如何。 +经过人工智能领域数十年的研究,一种名为 _Transformer_ 的新模型架构克服了 RNN 的限制,能够获得更长的文本序列作为输入。 Transformer 基于注意力机制,使模型能够为其接收到的输入赋予不同的权重,“更加专注于”关联信息集中的地方,不管它们在文本序列中的顺序如何。 生成式人工智能模也称为 LLM,因为它们使用文本输入和输出的架构。 这些模型的有趣之处在于,它们经过来自书籍、文章和网站等不同来源的大量未标记数据的训练,可以适应各种各样的任务,并以创造性的方式生成语法正确的文本。 因此,它们不仅极大地增强了机器“理解”输入文本的能力,而且使机器能够以人类语言生成原始响应。 @@ -71,50 +71,48 @@ 在下一章中,我们将探索不同类型的生成式 AI 模型,但现在让我们看看大型语言模型是如何工作的,重点是 OpenAI GPT(生成式预训练 Transformer)模型。 -* **分词器,文本到数字**:大型语言模型接收文本作为输入并生成文本作为输出。 然而,作为统计模型,它们对数字的处理效果对比起文本序列的处理效果要好得多。 这就是为什么模型的每个输入在被核心模型使用之前都由分词器处理。 标记是一段文本——由可变数量的字符组成,因此标记器的主要任务是将输入分割成标记数组。 然后,每个令牌都映射有一个令牌索引,该索引是原始文本块的整数编码。 +- **分词器,文本到数字**:大型语言模型接收文本作为输入并生成文本作为输出。 然而,作为统计模型,它们对数字的处理效果对比起文本序列的处理效果要好得多。 这就是为什么模型的每个输入在被核心模型使用之前都由分词器处理。 标记是一段文本——由可变数量的字符组成,因此标记器的主要任务是将输入分割成标记数组。 然后,每个令牌都映射有一个令牌索引,该索引是原始文本块的整数编码。 ![Example of tokenization](../../images/tokenizer-example.png?WT.mc_id=academic-105485-koreyst) -* **预测输出标记**:给定 n 个标记作为输入(最大 n 因模型而异),模型能够预测一个标记作为输出。 然后,该标记会以扩展窗口模式合并到下一次迭代的输入中,从而获得一个(或多个)句子作为答案,生成结果有更好的用户体验。 这解释了为什么如果您曾经使用过 ChatGPT,您可能会注意到有时它在生成结果时在句子中间出现停顿。 +- **预测输出标记**:给定 n 个标记作为输入(最大 n 因模型而异),模型能够预测一个标记作为输出。 然后,该标记会以扩展窗口模式合并到下一次迭代的输入中,从而获得一个(或多个)句子作为答案,生成结果有更好的用户体验。 这解释了为什么如果您曾经使用过 ChatGPT,您可能会注意到有时它在生成结果时在句子中间出现停顿。 -* **选择过程,概率分布**:模型根据其在当前文本序列之后出现的概率来选择输出标记。 这是因为该模型预测了根据其训练计算出的所有可能的“下一个标记”的概率分布。 然而,并不总是从结果分布中选择概率最高的标记。 这种选择增加了一定程度的随机性,模型以非确定性方式运行——对于相同的输入,我们不会得到完全相同的输出。 添加这种程度的随机性是为了模拟人类创造性思维的过程,您可以使用称为温度的模型参数进行调整。 +- **选择过程,概率分布**:模型根据其在当前文本序列之后出现的概率来选择输出标记。 这是因为该模型预测了根据其训练计算出的所有可能的“下一个标记”的概率分布。 然而,并不总是从结果分布中选择概率最高的标记。 这种选择增加了一定程度的随机性,模型以非确定性方式运行——对于相同的输入,我们不会得到完全相同的输出。 添加这种程度的随机性是为了模拟人类创造性思维的过程,您可以使用称为温度的模型参数进行调整。 ## “Our startup” 如何利用 LLMs ? -现在我们对 LLMs 的内部工作有了更好的了解,让我们看看它们可以很好地执行的最常见任务的一些实际示例,并着眼于我们的业务场景。 -我们说过, LLMs 的主要功能是*从头开始生成文本,从文本输入开始,用自然语言编写*。 +现在我们对 LLMs 的内部工作有了更好的了解,让我们看看它们可以很好地执行的最常见任务的一些实际示例,并着眼于我们的业务场景。 +我们说过, LLMs 的主要功能是*从头开始生成文本,从文本输入开始,用自然语言编写*。 但是是什么样的文本输入和输出呢? 大型语言模型的输入称为提示,而输出称为补全,术语指的是生成下一个标记来完成当前输入的模型机制。 我们将深入探讨什么是提示以及如何设计它以充分和我们的模型进行交流。 但现在,我们假设提示可能包括: -* 通过一条**指令**,指定我们期望模型输出的类型。 该指令有时可能会嵌入一些示例或一些附加数据。 +- 通过一条**指令**,指定我们期望模型输出的类型。 该指令有时可能会嵌入一些示例或一些附加数据。 - 1. 文章、书籍、产品评论等的总结,以及从非结构化数据中提取见解。 + 1. 文章、书籍、产品评论等的总结,以及从非结构化数据中提取见解。 - ![摘要示例](../../images/summarization-example.png?WT.mc_id=academic-105485-koreyst) + ![摘要示例](../../images/summarization-example.png?WT.mc_id=academic-105485-koreyst) + 2. 文章、论文、作业等的创意构思和设计。 - 2. 文章、论文、作业等的创意构思和设计。 + ![创意写作示例](../../images/creative-writing-example.png?WT.mc_id=academic-105485-koreyst) - ![创意写作示例](../../images/creative-writing-example.png?WT.mc_id=academic-105485-koreyst) +- **问题**,以与代理对话的形式提出。 -* **问题**,以与代理对话的形式提出。 + ![对话示例](../../images/conversation-example.png?WT.mc_id=academic-105485-koreyst) - ![对话示例](../../images/conversation-example.png?WT.mc_id=academic-105485-koreyst) +- **文本补全**,这隐含着对写作帮助的请求。 -* **文本补全**,这隐含着对写作帮助的请求。 + ![文本完成示例](../../images/text-completion-example.png?WT.mc_id=academic-105485-koreyst) - ![文本完成示例](../../images/text-completion-example.png?WT.mc_id=academic-105485-koreyst) +- **代码**解释和记录需求,或者要求生成执行特定任务的一段代码的注释。 -* **代码**解释和记录需求,或者要求生成执行特定任务的一段代码的注释。 - - ![编码示例](../../images/coding-example.png?WT.mc_id=academic-105485-koreyst) + ![编码示例](../../images/coding-example.png?WT.mc_id=academic-105485-koreyst) 以上的例子非常简单,并不是对生成式人工智能功能的详尽演示。 只是想展示使用生成式人工智能的潜力,并不局限于教育领域。 此外,生成式人工智能模型的输出并不完美,有时模型的创造力可能会对其产生不利影响,导致输出是人类用户可以将其解释为现实神秘化的单词组合,或者具有攻击性。 生成式人工智能并不智能——至少在更全面的智能定义中是这样,包括批判性和创造性推理或情商; 它不是确定性的,也不值得信赖,因为错误的引用、内容和陈述等幻觉可能会与正确的信息结合起来,并以有说服力和自信的方式呈现。 在接下来的课程中,我们将处理所有这些限制,并了解可以采取哪些措施来降低影响。 - ## 任务 你的任务是阅读更多关于生成式人工智能(https://en.wikipedia.org/wiki/Generative_artificial_intelligence)的内容,并尝试找出一个你现在可以添加生成式人工智能但目前还没有的领域。 与用“旧式方法”做有什么不同,你能做一些你以前做不到的事情吗,或者生成的时候速度更快吗? 写一篇 300 字的摘要,描述您梦想的人工智能初创公司的样子,包括“问题”、“我将如何使用人工智能”、“影响”等标题,还可以选择做一份商业计划。 diff --git a/01-introduction-to-genai/translations/es-mx/README.md b/01-introduction-to-genai/translations/es-mx/README.md index 2bb6efd30..1e53ee579 100644 --- a/01-introduction-to-genai/translations/es-mx/README.md +++ b/01-introduction-to-genai/translations/es-mx/README.md @@ -1,8 +1,8 @@ # 📚 Introducción a la Inteligencia Artificial Generativa y a los Modelos de Lenguaje Grandes -[![Introducción a la Inteligencia Artificial Generativa y a los Modelos de Lenguaje Grandes](../../images/01-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/vf_mZrn8ibc?WT.mc_id=academic-105485-koreyst) +[![Introducción a la Inteligencia Artificial Generativa y a los Modelos de Lenguaje Grandes](../../images/01-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=36c6795a-e63c-46dd-8d69-df8bbe6e7bc9?WT.mc_id=academic-105485-koreyst) -*(Haz clic en la imagen de arriba para ver el video de esta lección)* +_(Haz clic en la imagen de arriba para ver el video de esta lección)_ La Inteligencia Artificial Generativa es una forma de inteligencia artificial capaz de generar texto, imágenes y otros tipos de contenido. Lo que la hace una tecnología fantástica es que democratiza la IA; cualquiera puede utilizarla con tan solo una indicación de texto, una oración escrita en un lenguaje natural. No es necesario que aprendas un lenguaje como Java o SQL para lograr algo valioso; todo lo que necesitas es usar tu propio lenguaje, expresar lo que deseas, y obtendrás sugerencias de un modelo de IA. Las aplicaciones e impacto de esto son enormes: puedes redactar o entender informes, escribir aplicaciones y mucho más, todo en cuestión de segundos. @@ -12,23 +12,23 @@ En este plan de estudios, exploraremos cómo nuestra startup aprovecha la inteli Esta lección cubrirá: -* Introducción al escenario empresarial: nuestra idea y misión como startup. -* Inteligencia Artificial Generativa y cómo llegamos al panorama tecnológico actual. -* Funcionamiento interno de un modelo de lenguaje grande. -* Principales capacidades y casos de uso prácticos de los Modelos de Lenguaje Grandes. +- Introducción al escenario empresarial: nuestra idea y misión como startup. +- Inteligencia Artificial Generativa y cómo llegamos al panorama tecnológico actual. +- Funcionamiento interno de un modelo de lenguaje grande. +- Principales capacidades y casos de uso prácticos de los Modelos de Lenguaje Grandes. ## Objetivos de Aprendizaje Después de completar esta lección, comprenderás: -* Qué es la inteligencia artificial generativa y cómo funcionan los Modelos de Lenguaje Grandes. -* Cómo puedes aprovechar los modelos de lenguaje grandes para diferentes casos de uso, con un enfoque en escenarios educativos. +- Qué es la inteligencia artificial generativa y cómo funcionan los Modelos de Lenguaje Grandes. +- Cómo puedes aprovechar los modelos de lenguaje grandes para diferentes casos de uso, con un enfoque en escenarios educativos. -## Escenario: nuestra startup educativa +## Escenario: nuestra startup educativa La Inteligencia Artificial Generativa (IA) representa la cúspide de la tecnología de la IA, empujando los límites de lo que antes se consideraba imposible. Los modelos de IA generativa tienen varias capacidades y aplicaciones, pero en este plan de estudios exploraremos cómo está revolucionando la educación a través de una startup ficticia. Nos referiremos a esta startup como nuestra startup. Nuestra startup opera en el ámbito educativo con la ambiciosa declaración de misión de: -> *mejorar la accesibilidad en el aprendizaje a escala global, garantizando un acceso equitativo a la educación y proporcionando experiencias de aprendizaje personalizadas a cada estudiante, según sus necesidades.*. +> _mejorar la accesibilidad en el aprendizaje a escala global, garantizando un acceso equitativo a la educación y proporcionando experiencias de aprendizaje personalizadas a cada estudiante, según sus necesidades._. Nuestro equipo de startup es consciente de que no podremos alcanzar este objetivo sin aprovechar una de las herramientas más poderosas de los tiempos modernos: los Modelos de Lenguaje Grandes (LLMs). @@ -40,7 +40,7 @@ Para empezar, definamos algunos conceptos y terminología básicos que utilizare ## ¿Cómo llegamos a la Inteligencia Artificial Generativa? -A pesar del extraordinario *hype* creado recientemente por el anuncio de modelos de IA generativa, esta tecnología lleva décadas en desarrollo, con los primeros esfuerzos de investigación remontándose a los años 60. Ahora estamos en un punto en el que la IA tiene capacidades cognitivas humanas, como la conversación, como se muestra, por ejemplo [OpenAI ChatGPT](https://openai.com/chatgpt) o [Bing Chat](https://www.microsoft.com/edge/features/bing-chat?WT.mc_id=academic-105485-koreyst), que también utiliza un modelo GPT para las conversaciones de búsqueda web en Bing. +A pesar del extraordinario _hype_ creado recientemente por el anuncio de modelos de IA generativa, esta tecnología lleva décadas en desarrollo, con los primeros esfuerzos de investigación remontándose a los años 60. Ahora estamos en un punto en el que la IA tiene capacidades cognitivas humanas, como la conversación, como se muestra, por ejemplo [OpenAI ChatGPT](https://openai.com/chatgpt) o [Bing Chat](https://www.microsoft.com/edge/features/bing-chat?WT.mc_id=academic-105485-koreyst), que también utiliza un modelo GPT para las conversaciones de búsqueda web en Bing. Volviendo un poco atrás, los primeros prototipos de IA consistían en chatbots mecanografiados, que dependían de una base de conocimientos extraída de un grupo de expertos y representada en una computadora. Las respuestas en la base de conocimientos eran activadas por palabras clave que aparecían en el texto de entrada. Sin embargo, pronto quedó claro que este enfoque, utilizando chatbots mecanografiados, no escalaba bien. @@ -62,7 +62,7 @@ Así es como llegamos a la Inteligencia Artificial Generativa hoy en día, que s ![AI, ML, DL y IA Generativa](../../images/AI-diagram.png?WT.mc_id=academic-105485-koreyst) -Después de décadas de investigación en el campo de la inteligencia artificial, una nueva arquitectura de modelo, - llamada *Transformer* – superó los límites de las RNN, siendo capaz de manejar secuencias de texto mucho más largas como entrada. Los Transformers se basan en el mecanismo de atención, lo que permite al modelo asignar diferentes pesos a las entradas que recibe, ‘prestando más atención’ donde se concentra la información más relevante, independientemente de su orden en la secuencia de texto. +Después de décadas de investigación en el campo de la inteligencia artificial, una nueva arquitectura de modelo, - llamada _Transformer_ – superó los límites de las RNN, siendo capaz de manejar secuencias de texto mucho más largas como entrada. Los Transformers se basan en el mecanismo de atención, lo que permite al modelo asignar diferentes pesos a las entradas que recibe, ‘prestando más atención’ donde se concentra la información más relevante, independientemente de su orden en la secuencia de texto. La mayoría de los modelos recientes de Inteligencia Artificial Generativa, también conocidos como Modelos de Lenguaje Grandes (LLMs, por sus siglas en inglés), ya que trabajan con entradas y salidas de texto, están basados de hecho en esta arquitectura. Lo interesante de estos modelos, entrenados con una gran cantidad de datos no etiquetados de diversas fuentes como libros, artículos y sitios web, es que pueden adaptarse a una amplia variedad de tareas y generar texto gramaticalmente correcto con un atisbo de creatividad. Por lo tanto, no solo mejoraron de manera increíble la capacidad de una máquina para 'entender' un texto de entrada, sino que también habilitaron su capacidad para generar una respuesta original en lenguaje humano. @@ -70,26 +70,26 @@ La mayoría de los modelos recientes de Inteligencia Artificial Generativa, tamb En el próximo capítulo exploraremos diferentes tipos de modelos de Inteligencia Artificial Generativa, pero por ahora echemos un vistazo a cómo funcionan los grandes modelos de lenguaje, con un enfoque en los modelos de OpenAI GPT (Generative Pre-trained Transformer). -* **Tokenizador, texto a números**: Los grandes modelos de lenguaje reciben un texto como entrada y generan un texto como salida. Sin embargo, al ser modelos estadísticos, funcionan mucho mejor con números que con secuencias de texto. Es por eso que cada entrada al modelo se procesa mediante un tokenizador antes de ser utilizada por el modelo central. Un token es un fragmento de texto, que consiste en un número variable de caracteres. La tarea principal del tokenizador es dividir la entrada en un conjunto de tokens. Luego, cada token se asigna con un índice de token, que es la codificación entera del fragmento de texto original. +- **Tokenizador, texto a números**: Los grandes modelos de lenguaje reciben un texto como entrada y generan un texto como salida. Sin embargo, al ser modelos estadísticos, funcionan mucho mejor con números que con secuencias de texto. Es por eso que cada entrada al modelo se procesa mediante un tokenizador antes de ser utilizada por el modelo central. Un token es un fragmento de texto, que consiste en un número variable de caracteres. La tarea principal del tokenizador es dividir la entrada en un conjunto de tokens. Luego, cada token se asigna con un índice de token, que es la codificación entera del fragmento de texto original. ![Ejemplo de tokenización](../../images/tokenizer-example.png?WT.mc_id=academic-105485-koreyst) -* **Predicción de tokens de salida**: Dado un conjunto de n tokens como entrada (con un máximo n que varía de un modelo a otro), el modelo es capaz de predecir un token como salida. Este token se incorpora luego a la entrada de la siguiente iteración, en un patrón de ventana expansiva, lo que permite una mejor experiencia del usuario al obtener una (o varias) oraciones como respuesta. Esto explica por qué, si alguna vez has interactuado con ChatGPT, es posible que hayas notado que a veces parece detenerse en medio de una oración. +- **Predicción de tokens de salida**: Dado un conjunto de n tokens como entrada (con un máximo n que varía de un modelo a otro), el modelo es capaz de predecir un token como salida. Este token se incorpora luego a la entrada de la siguiente iteración, en un patrón de ventana expansiva, lo que permite una mejor experiencia del usuario al obtener una (o varias) oraciones como respuesta. Esto explica por qué, si alguna vez has interactuado con ChatGPT, es posible que hayas notado que a veces parece detenerse en medio de una oración. -* **Proceso de selección, distribución de probabilidad**: El token de salida es elegido por el modelo de acuerdo con su probabilidad de ocurrir después de la secuencia de texto actual. Esto se debe a que el modelo predice una distribución de probabilidad sobre todos los posibles 'próximos tokens', calculados en base a su entrenamiento. Sin embargo, no siempre se elige el token con la probabilidad más alta de la distribución resultante. Se añade un grado de aleatoriedad a esta elección, de manera que el modelo actúa de manera no determinista; no obtenemos la misma salida exacta para la misma entrada. Este grado de aleatoriedad se agrega para simular el proceso de pensamiento creativo y se puede ajustar utilizando un parámetro del modelo llamado temperatura. +- **Proceso de selección, distribución de probabilidad**: El token de salida es elegido por el modelo de acuerdo con su probabilidad de ocurrir después de la secuencia de texto actual. Esto se debe a que el modelo predice una distribución de probabilidad sobre todos los posibles 'próximos tokens', calculados en base a su entrenamiento. Sin embargo, no siempre se elige el token con la probabilidad más alta de la distribución resultante. Se añade un grado de aleatoriedad a esta elección, de manera que el modelo actúa de manera no determinista; no obtenemos la misma salida exacta para la misma entrada. Este grado de aleatoriedad se agrega para simular el proceso de pensamiento creativo y se puede ajustar utilizando un parámetro del modelo llamado temperatura. ## ¿Cómo puede nuestra startup aprovechar los grandes modelos de lenguaje? -Ahora que tenemos una mejor comprensión del funcionamiento interno de un modelo de lenguaje grande, veamos algunos ejemplos prácticos de las tareas más comunes que pueden realizar bastante bien, con atención a nuestro escenario empresarial. Dijimos que la capacidad principal de un Modelo de Lenguaje Grande es *generar texto desde cero, a partir de una entrada textual escrita en lenguaje natural*. +Ahora que tenemos una mejor comprensión del funcionamiento interno de un modelo de lenguaje grande, veamos algunos ejemplos prácticos de las tareas más comunes que pueden realizar bastante bien, con atención a nuestro escenario empresarial. Dijimos que la capacidad principal de un Modelo de Lenguaje Grande es _generar texto desde cero, a partir de una entrada textual escrita en lenguaje natural_. ¿Pero qué tipo de entrada y salida textual? La entrada de un gran modelo de lenguaje se conoce como "prompt" (indicación), mientras que la salida se conoce como "completion" (completado), término que se refiere al mecanismo del modelo de generar el próximo token para completar la entrada actual. Vamos a profundizar en lo que es un "prompt" y cómo diseñarlo de manera que aprovechemos al máximo nuestro modelo. Pero por ahora, simplemente diremos que un "prompt" puede incluir: -* Una **instrucción** especificando el tipo de salida que esperamos del modelo. Esta instrucción a veces puede incluir algunos ejemplos o datos adicionales. +- Una **instrucción** especificando el tipo de salida que esperamos del modelo. Esta instrucción a veces puede incluir algunos ejemplos o datos adicionales. - 1. Resumen de un artículo, libro, reseñas de productos y más, junto con la extracción de información clave a partir de datos no estructurados. - - ![Ejemplo de resumen](../../images/summarization-example.png?WT.mc_id=academic-105485-koreyst) + 1. Resumen de un artículo, libro, reseñas de productos y más, junto con la extracción de información clave a partir de datos no estructurados. + + ![Ejemplo de resumen](../../images/summarization-example.png?WT.mc_id=academic-105485-koreyst)
@@ -98,21 +98,22 @@ La entrada de un gran modelo de lenguaje se conoce como "prompt" (indicación), ![Ejemplo de escritura creativa](../../images/creative-writing-example.png?WT.mc_id=academic-105485-koreyst)
- -* Una **pregunta**, preguntado en forma de conversación con un agente. - + + +- Una **pregunta**, preguntado en forma de conversación con un agente. + ![Ejemplo de conversación](../../images/conversation-example.png?WT.mc_id=academic-105485-koreyst)
-* Un fragmento de **texto por completar**, lo cual implícitamente es una solicitud de ayuda en la escritura. - +- Un fragmento de **texto por completar**, lo cual implícitamente es una solicitud de ayuda en la escritura. + ![Ejemplo de finalización de texto](../../images/text-completion-example.png?WT.mc_id=academic-105485-koreyst)
-* Un trozo de **código** junto con la petición de explicarlo y documentarlo, o un comentario pidiendo generar un fragmento de código que realice una tarea específica. - +- Un trozo de **código** junto con la petición de explicarlo y documentarlo, o un comentario pidiendo generar un fragmento de código que realice una tarea específica. + ![Ejemplo de código](../../images/coding-example.png?WT.mc_id=academic-105485-koreyst)
@@ -134,7 +135,7 @@ Si realizas esta tarea, incluso podrías estar listo para postularte al programa 1. Obtienes la misma respuesta exacta cada vez. 2. Realiza tareas de manera perfecta, es excelente sumando números, produciendo código funcional, etc. 3. La respuesta puede variar a pesar de usar el mismo prompt. También es excelente para ofrecerte un primer borrador de algo, ya sea texto o código. Sin embargo, necesitas mejorar los resultados. - + A: 3. Un LLM (Large Language Model) es no determinista, las respuestas varían, sin embargo, puedes controlar su variabilidad mediante un ajuste de temperatura. Tampoco debes esperar que haga las cosas de manera perfecta; está aquí para hacer el trabajo pesado por ti, lo que a menudo significa que obtienes un buen primer intento en algo que necesitas mejorar gradualmente. ## ¡Gran trabajo! ¡Continúa el viaje! diff --git a/01-introduction-to-genai/translations/ja-jp/README.md b/01-introduction-to-genai/translations/ja-jp/README.md index e95506168..435add0aa 100644 --- a/01-introduction-to-genai/translations/ja-jp/README.md +++ b/01-introduction-to-genai/translations/ja-jp/README.md @@ -1,8 +1,8 @@ # 生成 AI と大規模言語モデルの紹介 -[![Introduction to Generative AI and Large Language Models](../../images/01-lesson-banner.png?WT.mc_id=academic-105485-yoterada)](https://youtu.be/vf_mZrn8ibc?WT.mc_id=academic-105485-yoterada) +[![Introduction to Generative AI and Large Language Models](../../images/01-lesson-banner.png?WT.mc_id=academic-105485-yoterada)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=36c6795a-e63c-46dd-8d69-df8bbe6e7bc9?WT.mc_id=academic-105485-yoterada) -> *(上記の画像をクリックすると、レッスン・ビデオを表示します)* +> _(上記の画像をクリックすると、レッスン・ビデオを表示します)_ 生成 AI は、テキストや画像などのコンテンツを生成する人工知能 (Artificial Intelligence) です。この技術が素晴らしいのは、AI を誰でも使えるようにしている所で、自然言語で書いた文章やテキスト・プロンプトだけで利用できます。 皆様にとって価値のある何らかの成果を生み出すために、Java や SQL のような言語を学ぶ必要はありません。普段使っている自分の言葉で要望を伝えるだけで、AI モデルから提案が返ってきます。この技術を使えば、報告書を書いたり、その内容を理解するためのアプリケーションが、わずか数秒でできるようになります。その可能性と影響力は計り知れません。 @@ -13,17 +13,17 @@ このレッスンでは、下記の内容について説明します。 -* ビジネスシナリオの紹介: スタートアップのアイデアとミッション -* 生成 AI と、技術的な歴史背景 -* 大規模言語モデルの内部動作 -* 大規模言語モデルの主な機能と実用的なユースケース +- ビジネスシナリオの紹介: スタートアップのアイデアとミッション +- 生成 AI と、技術的な歴史背景 +- 大規模言語モデルの内部動作 +- 大規模言語モデルの主な機能と実用的なユースケース ## 学習目標 このレッスンを修了すると、下記を理解できます: -* 生成 AI とは何か、そして大規模言語モデルの仕組みを理解する -* 教育シナリオに重点を置き、さまざまなユース・ケースで大規模言語モデルを活用する方法を理解する +- 生成 AI とは何か、そして大規模言語モデルの仕組みを理解する +- 教育シナリオに重点を置き、さまざまなユース・ケースで大規模言語モデルを活用する方法を理解する ## シナリオ: 教育関連事業をターゲットにするスタートアップ企業 @@ -31,7 +31,7 @@ 生成 AI モデルは様々な機能と用途で利用可能ですが、このカリキュラムでは、架空のスタートアップ企業を通じて教育分野でどのように革新していくのかを学んでいきます。この新興企業を「スタートアップ」と呼びます。 スタートアップは、教育の領域において下記の野心的な目標を持って事業を行っています。 -> *世界規模で学習の利便性を高め、教育に対する平等な機会を実現し、学習者一人ひとりの要望に応える個別学習体験を提供します。* +> _世界規模で学習の利便性を高め、教育に対する平等な機会を実現し、学習者一人ひとりの要望に応える個別学習体験を提供します。_ スタートアップチームは、大規模言語モデル(LLM)という現在最も強力なツールを使わなければ、上記の目標を達成するのが難しいと考えています。 @@ -45,16 +45,16 @@ 最近発表された、生成 AI モデルは大きな注目を集めていますが、この技術は過去数十年にわたる開発の歴史があり、初期の研究は 1960 年代までさかのぼります。そして今、AI 技術は [OpenAI の ChatGPT](https://openai.com/chatgpt) や [Bing Chat](https://www.microsoft.com/edge/features/bing-chat?WT.mc_id=academic-105485-yoterada) のように会話能力を有するなど、人間の認知能力を持つレベルにまで達しています。また、これらのチャットシステムは、Bing の Web 検索における対話でも GPT モデルを採用しています。 -AI の歴史を振り返ると、初期の AI のプロトタイプは、テキストの手入力した内容(打ち込んだ文字情報)に基づいて回答するチャットボットで、専門家集団から抽出した知識をベースに、それをコンピューターに表現していました。知識ベースによる回答は、入力テキスト内に出現するキーワードがトリガーになっていました。しかし、人間による手入力形式のチャットボットは、スケール・アウトが難しいのに気づきました。 +AI の歴史を振り返ると、初期の AI のプロトタイプは、テキストの手入力した内容(打ち込んだ文字情報)に基づいて回答するチャットボットで、専門家集団から抽出した知識をベースに、それをコンピューターに表現していました。知識ベースによる回答は、入力テキスト内に出現するキーワードがトリガーになっていました。しかし、人間による手入力形式のチャットボットは、スケール・アウトが難しいのに気づきました。 ### 統計学的手法を基にした AI 技術: 機械学習 -1990年代に入り、テキスト解析に統計学的手法を応用し、転換点が訪れました。 +1990 年代に入り、テキスト解析に統計学的手法を応用し、転換点が訪れました。 これにより、データからパターンを学習する新しいアルゴリズムが開発されました。これは「機械学習」として知られ、学習にプログラミングを必要としませんでした。この手法により、機械は人間の言語理解を再現できるようになります。テキストとラベルの組み合わせで訓練された統計モデルが、メッセージの意図を示す定義済みのラベルで、未知の入力テキストを分類できるようになります。 ### ニューラルネットワーク(神経回路網)と最新の仮想アシスタント -最近では、より大量のデータと、複雑な計算を処理できるハードウェアの進化が、AI 分野の研究をさらに促進し、ニューラル・ネットワークやディープラーニング・アルゴリズムと呼ばれる高度な機械学習アルゴリズムの開発につながりました。 +最近では、より大量のデータと、複雑な計算を処理できるハードウェアの進化が、AI 分野の研究をさらに促進し、ニューラル・ネットワークやディープラーニング・アルゴリズムと呼ばれる高度な機械学習アルゴリズムの開発につながりました。 ニューラルネットワーク(特に再帰型ニューラルネットワーク - RNN と呼ばれる)は、自然言語処理を飛躍的に進化させ、文章中に含まれる単語の意味を、文脈を考慮して評価し、より有益な方法でテキストの意図を表現できるようになりました。 @@ -62,25 +62,25 @@ AI の歴史を振り返ると、初期の AI のプロトタイプは、テキ ### 現在の生成 AI -このような経緯を経て、今日における生成 AI が誕生しました。これはディープラーニングのサブセットとして考えられます。 +このような経緯を経て、今日における生成 AI が誕生しました。これはディープラーニングのサブセットとして考えられます。 ![AI, ML, DL and Generative AI](../../images/AI-diagram.png?WT.mc_id=academic-105485-yoterada) -長年の AI 研究を経て、*トランスフォーマー* と呼ばれる新しいモデルのアーキテクチャが登場し、RNN の限界を越え、より長文テキストのシーケンスを入力として受け取れるようになりました。トランスフォーマーは、アテンションメカニズム(注意機構)を採用しており、モデルが、受け取った入力に対して異なる重み付けを行います。これにより、テキスト内の単語の並ぶ順番に関わらず、最も関連性の高い情報が集中している部分に「より多くの注意を払う」ようになります。 +長年の AI 研究を経て、_トランスフォーマー_ と呼ばれる新しいモデルのアーキテクチャが登場し、RNN の限界を越え、より長文テキストのシーケンスを入力として受け取れるようになりました。トランスフォーマーは、アテンションメカニズム(注意機構)を採用しており、モデルが、受け取った入力に対して異なる重み付けを行います。これにより、テキスト内の単語の並ぶ順番に関わらず、最も関連性の高い情報が集中している部分に「より多くの注意を払う」ようになります。 -最近の生成 AI モデルの多くは、文章の入出力を行う「大規模言語モデル(LLM)」とも呼ばれており、実際にこのアーキテクチャに基づいています。これらのモデルが注目される理由は、書籍、記事、ウェブサイトなど様々な情報源から、膨大なラベルなしデータとして訓練されているにも関わらず、創造性を備え文法も正しい文章を作り出すのが特筆すべき点です。したがって、これらのモデルは、機械が入力されたテキストを「理解」する能力を飛躍的に向上させただけでなく、人間の言葉で独自の回答を作り出す能力も実現しました。 +最近の生成 AI モデルの多くは、文章の入出力を行う「大規模言語モデル(LLM)」とも呼ばれており、実際にこのアーキテクチャに基づいています。これらのモデルが注目される理由は、書籍、記事、ウェブサイトなど様々な情報源から、膨大なラベルなしデータとして訓練されているにも関わらず、創造性を備え文法も正しい文章を作り出すのが特筆すべき点です。したがって、これらのモデルは、機械が入力されたテキストを「理解」する能力を飛躍的に向上させただけでなく、人間の言葉で独自の回答を作り出す能力も実現しました。 ## 大規模言語モデルの仕組みを教えてください -次の章では、さまざまな種類の生成 AI モデルについて詳しく説明しますが、その前に、OpenAI の GPT(Generative Pre-trained Transformer)モデルを中心に、大規模言語モデルがどのように機能するのかを確認しましょう。 +次の章では、さまざまな種類の生成 AI モデルについて詳しく説明しますが、その前に、OpenAI の GPT(Generative Pre-trained Transformer)モデルを中心に、大規模言語モデルがどのように機能するのかを確認しましょう。 -* **トークナイザー、テキストから数値へ**: 大規模言語モデルは、テキストを入力として受け取り、テキストを出力として生成します。ただし、統計モデルのため、テキスト・シーケンスよりも数値の方が、とてもうまく機能します。そこでモデルに対するすべての入力は、コア・モデルが使用する前に、トークナイザーで処理されます。トークンは文字の塊で、可変数の文字から構成されています。トークナイザーは、入力された文字列をトークン毎に分割し、トークン配列に格納する処理を行います。その後、各トークンはトークン・インデックスと結びつけられます。トークン・インデックスは元のテキストのチャンク(断片)を整数で符号化したものになります。 +- **トークナイザー、テキストから数値へ**: 大規模言語モデルは、テキストを入力として受け取り、テキストを出力として生成します。ただし、統計モデルのため、テキスト・シーケンスよりも数値の方が、とてもうまく機能します。そこでモデルに対するすべての入力は、コア・モデルが使用する前に、トークナイザーで処理されます。トークンは文字の塊で、可変数の文字から構成されています。トークナイザーは、入力された文字列をトークン毎に分割し、トークン配列に格納する処理を行います。その後、各トークンはトークン・インデックスと結びつけられます。トークン・インデックスは元のテキストのチャンク(断片)を整数で符号化したものになります。 ![トークン化の例](../../images/tokenizer-example.png?WT.mc_id=academic-105485-yoterada) -* **出力トークンの予測**: 入力として n 個のトークンを受け取ると(モデルによって n の最大値は異なります)、モデルは出力として 1 個のトークンを予測する力を持っています。この予測されたトークンは次の反復処理の入力に追加され、拡張ウィンドウ・パターンにより、利用者が 1 つ以上の文を回答として得られるようになります。これは、ChatGPT を試した時に、文の途中で時々処理が停止するような動きにつながる理由です。 +- **出力トークンの予測**: 入力として n 個のトークンを受け取ると(モデルによって n の最大値は異なります)、モデルは出力として 1 個のトークンを予測する力を持っています。この予測されたトークンは次の反復処理の入力に追加され、拡張ウィンドウ・パターンにより、利用者が 1 つ以上の文を回答として得られるようになります。これは、ChatGPT を試した時に、文の途中で時々処理が停止するような動きにつながる理由です。 -* **選考過程、確率分布**: モデルは、現在のテキストの順番の後ろに続く文字を、確率的に可能性の高い結果に基づいて出力トークンを選びます。これは、モデルが過去の訓練に基づいて、「次に出現するトークン」の各選択肢についてどれだけ確率が高いかを計算し、その全体の確率分布を推論しているからです。ただし、得られた分布から最も確率の高いトークンが必ず選ばれるとは限りません。この選択には、ある程度のランダム性が導入されており、モデルが決定的に振る舞わないようにしています。つまり同じ入力に対して必ずしも同じ出力結果が得られるわけではありません。このランダム性は、創造的な思考過程を模倣するために導入され、温度 (temperature) というパラメータで調整できます。 +- **選考過程、確率分布**: モデルは、現在のテキストの順番の後ろに続く文字を、確率的に可能性の高い結果に基づいて出力トークンを選びます。これは、モデルが過去の訓練に基づいて、「次に出現するトークン」の各選択肢についてどれだけ確率が高いかを計算し、その全体の確率分布を推論しているからです。ただし、得られた分布から最も確率の高いトークンが必ず選ばれるとは限りません。この選択には、ある程度のランダム性が導入されており、モデルが決定的に振る舞わないようにしています。つまり同じ入力に対して必ずしも同じ出力結果が得られるわけではありません。このランダム性は、創造的な思考過程を模倣するために導入され、温度 (temperature) というパラメータで調整できます。 ## スタートアップは、大規模言語モデルをどのように活用できるでしょうか? @@ -91,52 +91,57 @@ AI の歴史を振り返ると、初期の AI のプロトタイプは、テキ しかし、具体的にどのようなテキストの入力と出力を想定しているのでしょうか? 大規模言語モデルの入力は「プロンプト」と呼ばれ、出力は「コンプリーション」と呼ばれています。これは、モデルが現在の入力に続く次のトークンを生成する仕組みを表しています。これからプロンプトの意味や、モデルを最大限に活用するための設計方法について詳しく確認していきますが、とりあえず現時点では、プロンプトは以下のような要素が含まれるとお考えください: -* **プロンプト(指示)**:モデルに対して、期待する出力結果や、出力フォーマットの種類を指定する。この指示には、場合によっては具体例やその他のデータを追加で埋め込めます。 +- **プロンプト(指示)**:モデルに対して、期待する出力結果や、出力フォーマットの種類を指定する。この指示には、場合によっては具体例やその他のデータを追加で埋め込めます。 + + 1. 記事、書籍、製品レビューなどの要約と、非構造化データから知見(インサイト)の抽出 + + 訳者追記: + + - Can you please summarize the following text for me ? + - 以下のテキストの内容を要約していただけますか? + + ![Example of summarization](../../images/summarization-example.png?WT.mc_id=academic-105485-yoterada) - 1. 記事、書籍、製品レビューなどの要約と、非構造化データから知見(インサイト)の抽出 - - 訳者追記: - - Can you please summarize the following text for me ? - - 以下のテキストの内容を要約していただけますか? - - ![Example of summarization](../../images/summarization-example.png?WT.mc_id=academic-105485-yoterada) -
1. 記事、エッセイ、課題などのアイデアとデザインの作成 - 訳者追記: - - Please write an assignment for high school students including four open-ended questions about Louis XIV and his court. - - 高校生用の課題を作成します。ルイ14世と彼の宮廷について、自由に回答できる 4 つの質問を作成してください。 - - ![Example of creative writing](../../images/creative-writing-example.png?WT.mc_id=academic-105485-yoterada) - + 訳者追記: + + - Please write an assignment for high school students including four open-ended questions about Louis XIV and his court. + - 高校生用の課題を作成します。ルイ 14 世と彼の宮廷について、自由に回答できる 4 つの質問を作成してください。 + + ![Example of creative writing](../../images/creative-writing-example.png?WT.mc_id=academic-105485-yoterada) +
-* **質問**:エージェントに対して会話形式で質問を尋ねる +- **質問**:エージェントに対して会話形式で質問を尋ねる - 訳者追記: - - Who is Louis XIV and why he is an important historical character? - - ルイ14世はどういう人で、なぜ彼は歴史的に重要な人物なのですか? -![Example of conversation](../../images/conversation-example.png?WT.mc_id=academic-105485-yoterada) + 訳者追記: + - Who is Louis XIV and why he is an important historical character? + - ルイ14世はどういう人で、なぜ彼は歴史的に重要な人物なのですか? + + ![Example of conversation](../../images/conversation-example.png?WT.mc_id=academic-105485-yoterada)
-* **テキストの補完** : 続きを書く。文章を完成させるための執筆依頼をしています +- **テキストの補完** : 続きを書く。文章を完成させるための執筆依頼をしています + + 訳者追記: - 訳者追記: - - Louis XIV is an important historical character because - - ルイ14世が歴史的に重要な人物である理由は + - Louis XIV is an important historical character because + - ルイ 14 世が歴史的に重要な人物である理由は ![Example of text completion](../../images/text-completion-example.png?WT.mc_id=academic-105485-yoterada)
-* **プログラムコード**: プログラムに関する説明や、コード記述を求める依頼。または特定の作業を行うコード生成を依頼します。 +- **プログラムコード**: プログラムに関する説明や、コード記述を求める依頼。または特定の作業を行うコード生成を依頼します。 + + 訳者追記: - 訳者追記: - - A Python function to sum two numbers a and b - - 二つの数値 a と b を合計する Python の関数 + - A Python function to sum two numbers a and b + - 二つの数値 a と b を合計する Python の関数 ![Coding example](../../images/coding-example.png?WT.mc_id=academic-105485-yoterada) @@ -150,7 +155,7 @@ AI の歴史を振り返ると、初期の AI のプロトタイプは、テキ この課題では、「生成 AI」に関する知識を深め、まだ生成 AI が導入されていない領域を特定し、そこにどう取り込めるかを検討してください。 -従来の方法と比べて、どのような違いがあるでしょうか?これまでにできなかった処理が、実現可能になるか、作業の速度が上がるかをご検討ください。「問題点」「AIの活用方法」「導入による効果」といった見出しを用いて、皆様がこれから創業する理想の AI スタートアップを 300 文字でまとめてください。可能であれば、ビジネスプランも記述してください。 +従来の方法と比べて、どのような違いがあるでしょうか?これまでにできなかった処理が、実現可能になるか、作業の速度が上がるかをご検討ください。「問題点」「AI の活用方法」「導入による効果」といった見出しを用いて、皆様がこれから創業する理想の AI スタートアップを 300 文字でまとめてください。可能であれば、ビジネスプランも記述してください。 この課題を達成すれば、マイクロソフトのスタートアップ用インキュベータープログラム、[Microsoft for Startups Founders Hub](https://www.microsoft.com/startups?WT.mc_id=academic-105485-yoterada) への応募資格が得られるかもしれません。Azure や OpenAI の使用クレジット、メンタリング、その他多くの支援を提供していますので、詳細はウェブサイトをご覧ください。 diff --git a/01-introduction-to-genai/translations/pt-br/README.md b/01-introduction-to-genai/translations/pt-br/README.md index eff60d7af..18c02b3f8 100644 --- a/01-introduction-to-genai/translations/pt-br/README.md +++ b/01-introduction-to-genai/translations/pt-br/README.md @@ -1,8 +1,8 @@ # Introdução à Inteligência Artificial Generativa e Grandes Modelos de Linguagem -[![Introduction to Generative AI and Large Language Models](../../images/01-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/vf_mZrn8ibc?WT.mc_id=academic-105485-koreyst) +[![Introduction to Generative AI and Large Language Models](../../images/01-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=36c6795a-e63c-46dd-8d69-df8bbe6e7bc9?WT.mc_id=academic-105485-koreyst) -*(Clique na imagem acima para assistir ao vídeo desta lição)* +_(Clique na imagem acima para assistir ao vídeo desta lição)_ A Inteligência Artificial Generativa é uma forma de inteligência artificial capaz de gerar texto, imagens e outros tipos de conteúdo. O que a faz uma tecnologia fantástica é que ela democratiza a IA. Além disso, qualquer pessoa pode usá-la com apenas uma pequena frase, uma sentença escrita em uma linguagem natural. Não é necessário aprender uma linguagem como Java ou SQL para realizar algo significativo. Tudo o que você precisa fazer é usar sua própria linguagem, declarar o que deseja e uma sugestão de um modelo de IA será gerada. As aplicações e o impacto disso são enormes: você pode escrever ou compreender relatórios, criar aplicações e muito mais, tudo em questão de segundos. @@ -12,23 +12,23 @@ Neste currículo, exploraremos como nossa startup aproveita a IA generativa para Esta lição abordará: -* Introdução ao cenário de negócios: nossa ideia de startup e missão. -* Inteligência Artificial Generativa e como chegamos ao cenário tecnológico atual. -* Funcionamento interno de um grande modelo de linguagem. -* Principais capacidades e casos de uso práticos de Grandes Modelos de Linguagem. +- Introdução ao cenário de negócios: nossa ideia de startup e missão. +- Inteligência Artificial Generativa e como chegamos ao cenário tecnológico atual. +- Funcionamento interno de um grande modelo de linguagem. +- Principais capacidades e casos de uso práticos de Grandes Modelos de Linguagem. ## Objetivos de Aprendizado Após completar esta lição, você entenderá: -* O que é a Inteligência Artificial Generativa e como os Grandes Modelos de Linguagem funcionam. -* Como você pode aproveitar os Grandes Modelos de Linguagem para diferentes casos de uso, com foco em cenários de educação. +- O que é a Inteligência Artificial Generativa e como os Grandes Modelos de Linguagem funcionam. +- Como você pode aproveitar os Grandes Modelos de Linguagem para diferentes casos de uso, com foco em cenários de educação. ## Cenário: nossa startup educacional -A Inteligência Artificial Generativa (IA) representa o auge da tecnologia de IA, ultrapassando os limites do que antes era considerado impossível. Modelos de IA generativa têm várias capacidades e aplicações. Porém, neste currículo, exploraremos como ela está revolucionando a educação por meio de uma startup fictícia. Nos referiremos a essa startup como *nossa startup*. `Nossa startup` atua no domínio da educação com a ambiciosa declaração de missão: +A Inteligência Artificial Generativa (IA) representa o auge da tecnologia de IA, ultrapassando os limites do que antes era considerado impossível. Modelos de IA generativa têm várias capacidades e aplicações. Porém, neste currículo, exploraremos como ela está revolucionando a educação por meio de uma startup fictícia. Nos referiremos a essa startup como _nossa startup_. `Nossa startup` atua no domínio da educação com a ambiciosa declaração de missão: -> *Melhorar a acessibilidade na aprendizagem, em escala global, garantindo acesso equitativo à educação e proporcionando experiências de aprendizagem personalizadas a cada aluno(a), de acordo com suas necessidades*. +> _Melhorar a acessibilidade na aprendizagem, em escala global, garantindo acesso equitativo à educação e proporcionando experiências de aprendizagem personalizadas a cada aluno(a), de acordo com suas necessidades_. A equipe de nossa startup está ciente de que não será capaz de alcançar esse objetivo sem aproveitar uma das ferramentas mais poderosas dos tempos modernos - Grandes Modelos de Linguagem (LLMs). @@ -40,7 +40,7 @@ Para começar, vamos definir alguns conceitos e terminologia básica que usaremo ## Como surgiu a IA Generativa? -Apesar da extraordinária *hype* criada recentemente com o anúncio de modelos de IA generativa, essa tecnologia está em desenvolvimento há décadas, com os primeiros esforços de pesquisa datando dos anos 60. Agora estamos em um ponto em que a IA possui capacidades cognitivas humanas como conversação, como mostrado por exemplo, no [OpenAI ChatGPT](https://openai.com/chatgpt?WT.mc_id=academic-105485-koreyst) ou no [Bing Chat](https://www.microsoft.com/edge/features/bing-chat?WT.mc_id=academic-105485-koreyst), que também utiliza um modelo GPT para a pesquisa na web e conversas do Bing. +Apesar da extraordinária _hype_ criada recentemente com o anúncio de modelos de IA generativa, essa tecnologia está em desenvolvimento há décadas, com os primeiros esforços de pesquisa datando dos anos 60. Agora estamos em um ponto em que a IA possui capacidades cognitivas humanas como conversação, como mostrado por exemplo, no [OpenAI ChatGPT](https://openai.com/chatgpt?WT.mc_id=academic-105485-koreyst) ou no [Bing Chat](https://www.microsoft.com/edge/features/bing-chat?WT.mc_id=academic-105485-koreyst), que também utiliza um modelo GPT para a pesquisa na web e conversas do Bing. Voltando um pouco, os primeiros protótipos de IA consistiam em chatbots datilografados, baseando-se em uma base de conhecimento extraída de um grupo de especialistas e representada em um computador. As respostas na base de conhecimento eram acionadas por palavras-chave que apareciam no texto de entrada. No entanto, logo ficou claro que tal abordagem, usando chatbots datilografados, não se escalava bem. @@ -63,46 +63,46 @@ Foi assim que chegamos à IA Generativa de hoje, que pode ser vista como um subc ![IA, ML, AP e IA Generativa](../../images/AI-diagram.png?WT.mc_id=academic-105485-koreyst) -Depois de décadas de pesquisa no campo da IA, uma nova arquitetura de modelo - chamada *Transformer* - superou os limites das `RNNs`. Sendo capaz de receber sequências de texto muito mais longas como entrada. Os `Transformers` são baseados no mecanismo de atenção, permitindo que o modelo dê pesos diferentes às entradas que recebe, *prestando mais atenção* onde as informações mais relevantes estão concentradas, independentemente de sua ordem na sequência de texto. +Depois de décadas de pesquisa no campo da IA, uma nova arquitetura de modelo - chamada _Transformer_ - superou os limites das `RNNs`. Sendo capaz de receber sequências de texto muito mais longas como entrada. Os `Transformers` são baseados no mecanismo de atenção, permitindo que o modelo dê pesos diferentes às entradas que recebe, _prestando mais atenção_ onde as informações mais relevantes estão concentradas, independentemente de sua ordem na sequência de texto. -A maioria dos modelos recentes da IA generativa - também conhecidos como Grandes Modelos de Linguagem (LLMs), trabalham com entradas e saídas textuais e que, são de fato, baseados nessa arquitetura. O que é interessante sobre esses modelos - treinados em uma enorme quantidade de dados não rotulados de diversas fontes, como: livros, artigos e sites - é que eles podem ser adaptados para uma ampla variedade de tarefas e gerar texto gramaticalmente correto com uma semelhança de criatividade. Portanto, não apenas aumentaram incrivelmente a capacidade de uma máquina *entender* um texto de entrada. Mas também habilitaram sua capacidade de gerar uma resposta original em linguagem humana. +A maioria dos modelos recentes da IA generativa - também conhecidos como Grandes Modelos de Linguagem (LLMs), trabalham com entradas e saídas textuais e que, são de fato, baseados nessa arquitetura. O que é interessante sobre esses modelos - treinados em uma enorme quantidade de dados não rotulados de diversas fontes, como: livros, artigos e sites - é que eles podem ser adaptados para uma ampla variedade de tarefas e gerar texto gramaticalmente correto com uma semelhança de criatividade. Portanto, não apenas aumentaram incrivelmente a capacidade de uma máquina _entender_ um texto de entrada. Mas também habilitaram sua capacidade de gerar uma resposta original em linguagem humana. ## Como os Grandes Modelos de Linguagem funcionam? No próximo capítulo, vamos explorar diferentes tipos de modelos de IA generativa. Por enquanto, vamos dar uma olhada em como os grandes modelos de linguagem funcionam, com foco nos modelos do OpenAI GPT (`Generative Pre-trained Transformer`). -* **Tokenizador, texto para números**: Grandes Modelos de Linguagem, recebem um texto como entrada e geram um texto como saída. No entanto, sendo modelos estatísticos, eles funcionam muito melhor com números do que sequências de texto. É por isso que cada entrada no modelo é processada por um `tokenizador` antes de ser usada pelo modelo principal. Um `token` é um pedaço de texto - consistindo em um número variável de caracteres, portanto, a principal tarefa do tokenizador é dividir a entrada em uma matriz de tokens. Em seguida, cada token é mapeado com um índice de token, que é a codificação inteira do trecho de texto original. +- **Tokenizador, texto para números**: Grandes Modelos de Linguagem, recebem um texto como entrada e geram um texto como saída. No entanto, sendo modelos estatísticos, eles funcionam muito melhor com números do que sequências de texto. É por isso que cada entrada no modelo é processada por um `tokenizador` antes de ser usada pelo modelo principal. Um `token` é um pedaço de texto - consistindo em um número variável de caracteres, portanto, a principal tarefa do tokenizador é dividir a entrada em uma matriz de tokens. Em seguida, cada token é mapeado com um índice de token, que é a codificação inteira do trecho de texto original. ![Example of tokenization](../../images/tokenizer-example.png?WT.mc_id=academic-105485-koreyst) -* **Previsão de tokens de saída**: Dados `n` tokens como entrada (com o máximo `n` variando de um modelo para outro), o modelo é capaz de prever um token como saída. Esse token é então incorporado à entrada da próxima iteração em um padrão de janela expansiva. Possibilitando assim, uma melhor experiência do usuário em obter uma (ou várias) sentenças como resposta. Isso explica por que, se você já brincou com o ChatGPT, pode ter notado que às vezes parece que ele para no meio de uma frase. +- **Previsão de tokens de saída**: Dados `n` tokens como entrada (com o máximo `n` variando de um modelo para outro), o modelo é capaz de prever um token como saída. Esse token é então incorporado à entrada da próxima iteração em um padrão de janela expansiva. Possibilitando assim, uma melhor experiência do usuário em obter uma (ou várias) sentenças como resposta. Isso explica por que, se você já brincou com o ChatGPT, pode ter notado que às vezes parece que ele para no meio de uma frase. -* **Processo de seleção, distribuição de probabilidade**: O token de saída é escolhido pelo modelo de acordo com a probabilidade de ocorrer após a sequência de texto atual. Isso ocorre porque o modelo prevê uma distribuição de probabilidade sobre todos os possíveis *próximos tokens* calculados com base em seu treinamento. No entanto, nem sempre o token com a maior probabilidade é escolhido na distribuição resultante. Um grau de aleatoriedade é adicionado a essa escolha, de forma que o modelo age de maneira não determinística - não obtemos a mesma saída exata para a mesma entrada. Esse grau de aleatoriedade é adicionado para simular o processo de pensamento criativo e pode ser ajustado usando um parâmetro do modelo chamado temperatura. +- **Processo de seleção, distribuição de probabilidade**: O token de saída é escolhido pelo modelo de acordo com a probabilidade de ocorrer após a sequência de texto atual. Isso ocorre porque o modelo prevê uma distribuição de probabilidade sobre todos os possíveis _próximos tokens_ calculados com base em seu treinamento. No entanto, nem sempre o token com a maior probabilidade é escolhido na distribuição resultante. Um grau de aleatoriedade é adicionado a essa escolha, de forma que o modelo age de maneira não determinística - não obtemos a mesma saída exata para a mesma entrada. Esse grau de aleatoriedade é adicionado para simular o processo de pensamento criativo e pode ser ajustado usando um parâmetro do modelo chamado temperatura. ## Como nossa startup pode aproveitar os Grandes Modelos de Linguagem? Agora que temos uma melhor compreensão de como os grandes modelos de linguagem funcionam, vamos ver alguns exemplos práticos das tarefas mais comuns que eles podem realizar muito bem, com foco em nosso cenário de negócios. -Dissemos que a principal capacidade de um Grande Modelo de Linguagem é: *gerar um texto do zero, a partir de uma entrada textual, escrita em linguagem natural*. +Dissemos que a principal capacidade de um Grande Modelo de Linguagem é: _gerar um texto do zero, a partir de uma entrada textual, escrita em linguagem natural_. Mas que tipo de entrada e saída textual? A entrada de um grande modelo de linguagem é conhecida como `prompt`. Enquanto a saída é conhecida como `completion`, termo que se refere ao mecanismo do modelo de gerar o próximo token para completar a entrada atual. Vamos nos aprofundar no que é um prompt e como projetá-lo de maneira a obter o máximo de nosso modelo. Agora, vamos apenas dizer que um prompt pode incluir: -* Uma **instrução** especificando o tipo de saída que esperamos do modelo. Essa instrução às vezes pode incorporar alguns exemplos ou alguns dados adicionais. +- Uma **instrução** especificando o tipo de saída que esperamos do modelo. Essa instrução às vezes pode incorporar alguns exemplos ou alguns dados adicionais. - 1. Resumo de um artigo, livro, análises de produtos e muito mais. Juntamente com a extração de insights de dados não estruturados. - ![Exemplo de resumo](../../images/summarization-example.png?WT.mc_id=academic-105485-koreyst) + 1. Resumo de um artigo, livro, análises de produtos e muito mais. Juntamente com a extração de insights de dados não estruturados. + ![Exemplo de resumo](../../images/summarization-example.png?WT.mc_id=academic-105485-koreyst) - 2. Ideação criativa e design de um artigo, uma redação, uma tarefa ou mais. - ![Exemplo de escrita criativa](../../images/creative-writing-example.png?WT.mc_id=academic-105485-koreyst) + 2. Ideação criativa e design de um artigo, uma redação, uma tarefa ou mais. + ![Exemplo de escrita criativa](../../images/creative-writing-example.png?WT.mc_id=academic-105485-koreyst) -* Uma **pergunta** feita na forma de uma conversa com um agente. - ![Exemplo de conversa](../../images/conversation-example.png?WT.mc_id=academic-105485-koreyst) +- Uma **pergunta** feita na forma de uma conversa com um agente. + ![Exemplo de conversa](../../images/conversation-example.png?WT.mc_id=academic-105485-koreyst) -* Um trecho de **texto a ser completado** que implicitamente é um pedido de assistência na escrita. - ![Exemplo de conclusão de texto](../../images/text-completion-example.png?WT.mc_id=academic-105485-koreyst) +- Um trecho de **texto a ser completado** que implicitamente é um pedido de assistência na escrita. + ![Exemplo de conclusão de texto](../../images/text-completion-example.png?WT.mc_id=academic-105485-koreyst) -* Um trecho de **código** juntamente com a solicitação de explicá-lo e documentá-lo ou um comentário pedindo para gerar um trecho de código que execute uma tarefa específica. - ![Exemplo de programação](../../images/coding-example.png?WT.mc_id=academic-105485-koreyst) +- Um trecho de **código** juntamente com a solicitação de explicá-lo e documentá-lo ou um comentário pedindo para gerar um trecho de código que execute uma tarefa específica. + ![Exemplo de programação](../../images/coding-example.png?WT.mc_id=academic-105485-koreyst) Os exemplos acima são bastante simples e não pretendem ser uma demonstração exaustiva das capacidades dos Grandes Modelos de Linguagem. Eles apenas querem mostrar o potencial de uso da IA generativa, em particular, mas não limitado ao contexto educacional. @@ -112,9 +112,9 @@ Além disso, a saída de um modelo de IA generativa não é perfeita e, às veze Sua tarefa é estudar mais sobre [IA Generativa](https://en.wikipedia.org/wiki/Generative_artificial_intelligence?WT.mc_id=academic-105485-koreyst) e tentar identificar uma área onde você adicionaria IA Generativa hoje mas que ainda não a possui. Como o impacto seria diferente de fazer as coisas do "jeito antigo"? Você pode fazer algo que não era possível antes ou que seria mais rápido? Escreva um resumo de 300 palavras sobre como sua startup da IA dos sonhos seria e inclua cabeçalhos como: -- "Problema" -- "Como eu usaria a IA" -- "Impacto" +- "Problema" +- "Como eu usaria a IA" +- "Impacto" - E, como opção, um plano de negócios. Se você fizer essa tarefa, poderá até estar pronto para se inscrever no [Microsoft's Incubator](https://www.microsoft.com/startups?WT.mc_id=academic-105485-koreyst) para fundadores de startups. Eles oferecem créditos para Azure, OpenAI, mentoria e muito mais. Confira agora mesmo! @@ -127,7 +127,7 @@ O que é verdade sobre Grandes Modelos de Linguagem? 2. Eles fazem as coisas perfeitamente. São ótimos em somar números, produzir código funcional etc. 3. A resposta pode variar, mesmo ao usar o mesmo prompt. Além disso, eles são ótimos em fornecer um primeiro rascunho de algo, seja texto ou código. Mas você precisa melhorar os resultados. -**Resposta:** 3, um `LLM` é *não-determinístico*. A resposta varia, em você controlar sua variação por meio de uma configuração de `temperatura`. Além disso, você não deve esperar que ele faça as coisas perfeitamente. Ele está aqui para fazer o trabalho pesado para você, o que muitas vezes significa que você obtém uma boa primeira tentativa de algo que precisa melhorar gradualmente. +**Resposta:** 3, um `LLM` é _não-determinístico_. A resposta varia, em você controlar sua variação por meio de uma configuração de `temperatura`. Além disso, você não deve esperar que ele faça as coisas perfeitamente. Ele está aqui para fazer o trabalho pesado para você, o que muitas vezes significa que você obtém uma boa primeira tentativa de algo que precisa melhorar gradualmente. ## Ótimo trabalho! A Jornada Continua diff --git a/02-exploring-and-comparing-different-llms/translations/cn/README.md b/02-exploring-and-comparing-different-llms/translations/cn/README.md index 2e02761cd..9d29a50d2 100644 --- a/02-exploring-and-comparing-different-llms/translations/cn/README.md +++ b/02-exploring-and-comparing-different-llms/translations/cn/README.md @@ -1,8 +1,8 @@ # 第二章 : 探索和比较不同的 LLMs -[![Exploring and comparing different LLMs](../../images/02-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/J1mWzw0P74c?WT.mc_id=academic-105485-koreyst) +[![Exploring and comparing different LLMs](../../images/02-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=39aa0f98-826a-4f71-a24d-e888a8e80246?WT.mc_id=academic-105485-koreyst) -> *点击该图片看本章导学视频* +> _点击该图片看本章导学视频_ 在上一章中,我们了解了生成式人工智能如何改变技术格局、LLMs 工作原理,以及企业(例如“Our startup”)如何将它们应用到自己的应用场景中并得到发展! 在本章中,我们将比较和对比不同类型的大型语言模型以了解它们的优缺点。 @@ -80,7 +80,7 @@ LLMs 还可以根据其产生的输出进行分类。 文本或代码生成模型主要是生成文本或代码。 这些模型通常用于文本摘要、翻译和问答。 文本生成模型通常在大型文本数据集上进行训练,例如 [BookCorpus](https://www.cv-foundation.org/openaccess/content_iccv_2015/html/Zhu_Aligning_Books_and_ICCV_2015_paper.html),并且可用于生成新文本或回答问题。 代码生成模型,例如 [CodeParrot](https://huggingface.co/codeparrot?WT.mc_id=academic-105485-koreyst),通常在大型代码数据集(如 GitHub)上进行训练,可用于生成新代码或修复现有代码中的错误。 - ![Text and code generation](../../images/Text.png?WT.mc_id=academic-105485-koreyst) +![Text and code generation](../../images/Text.png?WT.mc_id=academic-105485-koreyst) ### 了解编码-解码器与独立解码器 @@ -92,13 +92,13 @@ LLMs 还可以根据其产生的输出进行分类。 审阅者就像一个独立编码器的模型,他们查看编写的课程和答案,注意它们之间的关系并通过上下文进行理解,但他们不擅于生成内容。独立编码器模型的一个例子是 BERT。 -想象一下,我们也可以有人可以创建和审查测验,这是一个编码器-解码器模型如 BART 和 T5 +想象一下,我们也可以有人可以创建和审查测验,这是一个编码器-解码器模型如 BART 和 T5 ### 理解服务与模型 现在,我们来谈谈服务和模型之间的区别。 服务是云服务提供商提供的产品,通常是模型、数据和其他组件的结合。 模型是服务的核心组件,通常是基础模型,例如各种 LLM。 -服务通常针对生产环境进行了优化,并且通常比模型让用户通过图形界面使用。 但服务并不总是免费提供的,可能需要订阅或付费才能使用,以换取服务所有者的设备和相关资源,优化费用并轻松扩展。 服务的一个例子是 [Azure OpenAI 服务](https://learn.microsoft.com/azure/ai-services/openai/overview?WT.mc_id=academic-105485-koreyst),它提供按需付费计划,意味着用户根据服务用量付费。此外,Azure OpenAI Service 在模型功能上还提供企业级安全性和负责任的 AI 框架。 +服务通常针对生产环境进行了优化,并且通常比模型让用户通过图形界面使用。 但服务并不总是免费提供的,可能需要订阅或付费才能使用,以换取服务所有者的设备和相关资源,优化费用并轻松扩展。 服务的一个例子是 [Azure OpenAI 服务](https://learn.microsoft.com/azure/ai-services/openai/overview?WT.mc_id=academic-105485-koreyst),它提供按需付费计划,意味着用户根据服务用量付费。此外,Azure OpenAI Service 在模型功能上还提供企业级安全性和负责任的 AI 框架。 模型是带有参数、权重等的神经网络。 允许公司在本地运营需要购买设备、构建并购买许可证或使用开源模型。 像 LLaMA 这样的模型使用就需要额外的算力运行 @@ -144,7 +144,7 @@ LLMs 还可以根据其产生的输出进行分类。 ![LLMs deployment](../../images/Deploy.png?WT.mc_id=academic-105485-koreyst) -图片来源: [企业部署LLM的四种方式| Fiddler AI 博客](https://www.fiddler.ai/blog/four-ways-that-enterprises-deploy-llms?WT.mc_id=academic-105485-koreyst) +图片来源: [企业部署 LLM 的四种方式| Fiddler AI 博客](https://www.fiddler.ai/blog/four-ways-that-enterprises-deploy-llms?WT.mc_id=academic-105485-koreyst) ### 通过上下文的提示工程 @@ -177,8 +177,6 @@ LLMs 有一个限制,即他们只能使用训练期间使用过的数据来生 从头开始培训 LLMs 无疑是最困难、最复杂的方法,需要大量数据、熟练资源和适当的计算能力。 仅在企业具有特定领域的用例和大量以特定领域为中心的数据的情况下才应考虑此选项。 - - ## 知识检查 提升 LLMs 输出效率最好的方法是什么? diff --git a/02-exploring-and-comparing-different-llms/translations/ja-jp/README.md b/02-exploring-and-comparing-different-llms/translations/ja-jp/README.md index d704da6f2..1e18ebf06 100644 --- a/02-exploring-and-comparing-different-llms/translations/ja-jp/README.md +++ b/02-exploring-and-comparing-different-llms/translations/ja-jp/README.md @@ -1,8 +1,8 @@ # さまざまな LLM の調査と比較 -[![Exploring and comparing different LLMs](../../images/02-lesson-banner.png?WT.mc_id=academic-105485-yoterada)](https://youtu.be/J1mWzw0P74c?WT.mc_id=academic-105485-yoterada) +[![Exploring and comparing different LLMs](../../images/02-lesson-banner.png?WT.mc_id=academic-105485-yoterada)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=36c6795a-e63c-46dd-8d69-df8bbe6e7bc9?WT.mc_id=academic-105485-yoterada) -> *(上記の画像をクリックすると、レッスン・ビデオを表示します)* +> _(上記の画像をクリックすると、レッスン・ビデオを表示します)_ 前回のレッスンで、生成 AI がどのようにして進化し、大規模言語モデル(LLM)がどのように機能するのか、そしてスタートアップがそれらを自分たちの目的に対して、どのように適用し成長できるかを見てきました。この章では、様々な大規模言語モデル(LLM)を比較し、それぞれの利点と欠点を理解していきます。 @@ -26,15 +26,15 @@ ## 大規模言語モデル(LLM)の様々な種類の理解 -大規模言語モデル(LLM)は、そのアーキテクチャ、トレーニングデータ、用途に応じて、様々なカテゴリに分類できます。こうした各モデルの把握は、スタートアップがシナリオに応じて最適なモデルを選択し、パフォーマンスのテストを行い、反復しながら改善するのに役立ちます。 +大規模言語モデル(LLM)は、そのアーキテクチャ、トレーニングデータ、用途に応じて、様々なカテゴリに分類できます。こうした各モデルの把握は、スタートアップがシナリオに応じて最適なモデルを選択し、パフォーマンスのテストを行い、反復しながら改善するのに役立ちます。 -LLM モデルには、さまざまな種類があり、どのモデルを選択するかは、それらを使用する目的、扱うデータ、ご利用可能な金額などによって異なります。 +LLM モデルには、さまざまな種類があり、どのモデルを選択するかは、それらを使用する目的、扱うデータ、ご利用可能な金額などによって異なります。 選択するモデルは、テキスト、オーディオ、ビデオ、画像の生成など、各用途に応じてそれぞれ異なる種類のモデルを選択します。 - **オーディオおよび音声認識**:この用途には、音声認識として汎用性のある Whisper というモデルが最適です。このモデルは、様々なオーディオ・データでトレーニングされており、多言語にも対応した音声認識ができます。[Whisper のモデルについての詳細はこちら](https://platform.openai.com/docs/models/whisper?WT.mc_id=academic-105485-yoterada). -- **画像生成**:画像生成の選択肢としては、DALL-E と Midjourney が非常に有名です。Azure OpenAI 上で DALL-E のモデルが利用可能です。[DALL-Eについての詳細はこちら](https://platform.openai.com/docs/models/dall-e?WT.mc_id=academic-105485-yoterada)をご覧ください。このカリキュラムの第 9 章でも解説します。 +- **画像生成**:画像生成の選択肢としては、DALL-E と Midjourney が非常に有名です。Azure OpenAI 上で DALL-E のモデルが利用可能です。[DALL-E についての詳細はこちら](https://platform.openai.com/docs/models/dall-e?WT.mc_id=academic-105485-yoterada)をご覧ください。このカリキュラムの第 9 章でも解説します。 - **テキスト生成**:多くのモデルがテキスト生成用にトレーニングされており、GPT-3.5 から GPT-4 に至るまで、多種多様な選択肢があります。それぞれ利用する際のコストは異なり、GPT-4 が最も高価です。機能とコストの観点から、ニーズに最も適したモデルを選択するために、[Azure OpenAI プレイグラウンド](https://oai.azure.com/portal/playground?WT.mc_id=academic-105485-yoterada)をご確認ください。 @@ -50,7 +50,7 @@ LLM モデルには、さまざまな種類があり、どのモデルを選択 ![ファウンデーション・モデルと LLM](../../images/FoundationModel.png?WT.mc_id=academic-105485-yoterada) -画像出展: [基盤モデルと大規模言語モデルの効果的なガイド | Babar M Bhatti著 | Medium](https://thebabar.medium.com/essential-guide-to-foundation-models-and-large-language-models-27dab58f7404) +画像出展: [基盤モデルと大規模言語モデルの効果的なガイド | Babar M Bhatti 著 | Medium](https://thebabar.medium.com/essential-guide-to-foundation-models-and-large-language-models-27dab58f7404) この違いをさらに明確にするために、ChatGPT を例に挙げてみましょう。ChatGPT の最初のバージョンを作る際 GPT-3.5 というモデルをファウンデーション・モデルとして使いました。つまり、OpenAI は、チャットボットのような会話シナリオで高いパフォーマンスを発揮するよう、チャットに特化したデータを用いて GPT-3.5 のチューニング版を作成したのです。 @@ -58,17 +58,17 @@ LLM モデルには、さまざまな種類があり、どのモデルを選択 画像出展: [2108.07258.pdf (arxiv.org)](https://arxiv.org/pdf/2108.07258.pdf?WT.mc_id=academic-105485-yoterada) -### オープンソース・モデルとプロプライエタリ・モデル +### オープンソース・モデルとプロプライエタリ・モデル -大規模言語モデル(LLM)を分類する別の方法として、それがオープンソースなのか、もしくはプロプライエタリな物なのか、という観点もあります。 +大規模言語モデル(LLM)を分類する別の方法として、それがオープンソースなのか、もしくはプロプライエタリな物なのか、という観点もあります。 オープンソース・モデルは、一般に公開され、誰でも利用できるモデルです。これらは多くの場合、そのモデルを開発した企業や研究コミュニティによって提供されます。これらのモデルは、LLM の様々な用途に合わせて検証、変更、カスタマイズの許可がされています。しかし、常に本番環境での利用に最適化されているわけではなく、プロプライエタリモデルほど高いパフォーマンスを発揮しない場合もあります。さらに、オープンソース・モデルの資金調達は限られており、長期的に継続できない可能性や、最新の研究に基づいて更新されていない可能性もあります。[Alpaca](https://crfm.stanford.edu/2023/03/13/alpaca.html?WT.mc_id=academic-105485-yoterada)、[Bloom](https://sapling.ai/llm/bloom?WT.mc_id=academic-105485-yoterada)、[LLaMA](https://sapling.ai/llm/llama?WT.mc_id=academic-105485-yoterada) などが人気のオープンソース・モデルの例です。 プロプライエタリ・モデルは、企業が所有し一般には公開されていないモデルです。これらのモデルは、通常本番環境での利用に最適化されています。しかし異なるユースケースに対して、検証、変更、カスタマイズは許可されていません。また、常に無料で利用できるわけではなく、利用するためには、サブスクリプション等による支払いが必要な場合もあります。さらに、利用者はモデルをトレーニングする際に使用するデータをコントロールできず、データのプライバシーや、責任ある AI の原則に基づく使用をモデル・プロバイダが保証しているのを信用しなければなりません。[OpenAI のモデル](https://platform.openai.com/docs/models/overview?WT.mc_id=academic-105485-yoterada)、[Google Bard](https://sapling.ai/llm/bard?WT.mc_id=academic-105485-yoterada)、[Claude 2](https://www.anthropic.com/index/claude-2?WT.mc_id=academic-105485-yoterada) などが人気のプロプライエタリ・モデルです。 -### 埋め込み (Embedding) と画像生成とテキスト・コード生成 +### 埋め込み (Embedding) と画像生成とテキスト・コード生成 -大規模言語モデル(LLM) は出力の種類によっても分類できます。 +大規模言語モデル(LLM) は出力の種類によっても分類できます。 埋め込み (Embedding) は、テキストを、「埋め込みと」呼ぶ数値形式に変換する AI モデルです。言い換えるならば、埋め込みは入力されたテキストに対する数値表現です。埋め込みによって、機械が単語や文の関係を理解しやすくなり、分類モデルや、数値データでパフォーマンスが向上するクラスタリング・モデルなど、他のモデルの入力として利用できます。埋め込みモデルは、たくさんのデータがある代理タスク用にモデルが作成され、モデルの重み(埋め込み)を、他の下流タスクで再利用する転移学習によく使用されます。このカテゴリーの例としては、[OpenAI Embeddings](https://platform.openai.com/docs/models/embeddings?WT.mc_id=academic-105485-yoterada) モデルがあります。 @@ -79,33 +79,33 @@ LLM モデルには、さまざまな種類があり、どのモデルを選択 ![Image generation](../../images/Image.png?WT.mc_id=academic-105485-yoterada) テキスト生成モデルとコード生成モデルは、テキストやコードを生成するためのモデルです。これらのモデルは、テキストの要約、翻訳、質疑応答などによく利用されます。テキスト生成モデルは、[BookCorpus](https://www.cv-foundation.org/openaccess/content_iccv_2015/html/Zhu_Aligning_Books_and_ICCV_2015_paper.html?WT.mc_id=academic-105485-yoterada) などの大規模なテキストデータセットでトレーニングされ、新しいテキストを生成したり、質問に答えたりするのに使われます。[CodeParrot](https://huggingface.co/codeparrot?WT.mc_id=academic-105485-yoterada) のようなコード生成モデルは、GitHub など大量のソースコード・データでトレーニングされ、新しいコードを生成したり、既存コードのバグ修正などで使われます。 - ![Text and code generation](../../images/Text.png?WT.mc_id=academic-105485-yoterada) +![Text and code generation](../../images/Text.png?WT.mc_id=academic-105485-yoterada) ### 「エンコーダー・デコーダー」 と 「デコーダーのみ」 のモデル -大規模言語モデル (LLM) のアーキテクチャの種類の違いについて解説するため、下記の例え話しを使います。 +大規模言語モデル (LLM) のアーキテクチャの種類の違いについて解説するため、下記の例え話しを使います。 -上司から、学生用のクイズを作成する仕事を任されたのを想像してください。あなたには 2 人の同僚がいて、1 人はコンテンツの作成を担当し、もう 1 人はそれをレビューする役割を担います。 +上司から、学生用のクイズを作成する仕事を任されたのを想像してください。あなたには 2 人の同僚がいて、1 人はコンテンツの作成を担当し、もう 1 人はそれをレビューする役割を担います。 -コンテンツを作成する人は「デコーダーのみ」のモデルに似ています。コンテンツ作成者は、トピックを見て、既に書いた内容を参考に、それに基づいてコンテンツを作成します。コンテンツ作成者は、魅力的で情報豊かなコンテンツを作成するのが得意ですが、トピックや学習目標を理解するのは得意ではありません。「デコーダーのみ」のモデルの例には、GPT-3 などの GPT ファミリーのモデルがあります。 +コンテンツを作成する人は「デコーダーのみ」のモデルに似ています。コンテンツ作成者は、トピックを見て、既に書いた内容を参考に、それに基づいてコンテンツを作成します。コンテンツ作成者は、魅力的で情報豊かなコンテンツを作成するのが得意ですが、トピックや学習目標を理解するのは得意ではありません。「デコーダーのみ」のモデルの例には、GPT-3 などの GPT ファミリーのモデルがあります。 -一方で、レビュー担当者は「エンコーダーのみ」のモデルに似ています。レビュー担当者は書かれた内容と回答を見て、それらの関係を把握し、文脈を理解しますが、コンテンツを作成するのは得意ではありません。「エンコーダーのみ」のモデルの例には、BERT があります。 +一方で、レビュー担当者は「エンコーダーのみ」のモデルに似ています。レビュー担当者は書かれた内容と回答を見て、それらの関係を把握し、文脈を理解しますが、コンテンツを作成するのは得意ではありません。「エンコーダーのみ」のモデルの例には、BERT があります。 -クイズを作成し、それをレビューする同一人物がいるのを想像してみてください。これが「エンコーダー・デコーダー」モデルです。BARTやT5などが例として挙げられます。 +クイズを作成し、それをレビューする同一人物がいるのを想像してみてください。これが「エンコーダー・デコーダー」モデルです。BART や T5 などが例として挙げられます。 ### サービスとモデル -サービスとモデルの違いについて説明します。サービスはクラウド・サービス・プロバイダーが提供する製品で、モデル、データ、その他のコンポーネントを組み合わせたものです。モデルはサービスの核となる部分で、大規模言語モデル (LLM) のようなファウンデーション・モデルが一般的です。 +サービスとモデルの違いについて説明します。サービスはクラウド・サービス・プロバイダーが提供する製品で、モデル、データ、その他のコンポーネントを組み合わせたものです。モデルはサービスの核となる部分で、大規模言語モデル (LLM) のようなファウンデーション・モデルが一般的です。 -サービスは、本番環境での利用に最適化されており、グラフィカル・ユーザー・インターフェースを通じて、直接モデルを操作するのに比べ扱いやすくなっています。しかしサービスは、常に無料で利用できるわけではなく、サービス提供者の機器やリソースを活用する代わりに、サブスクリプション等による支払いが必要な場合があります。これにより、費用を最適化し、簡単にスケールアップできます。サービスの例としては、使用量に応じて料金が発生する [Azure OpenAI Service](https://learn.microsoft.com/azure/ai-services/openai/overview?WT.mc_id=academic-105485-yoterada) があります。また、Azure OpenAI サービスは、モデルの能力に加え、エンタープライズグレードのセキュリティと責任ある AI のフレームワークも提供しています。 +サービスは、本番環境での利用に最適化されており、グラフィカル・ユーザー・インターフェースを通じて、直接モデルを操作するのに比べ扱いやすくなっています。しかしサービスは、常に無料で利用できるわけではなく、サービス提供者の機器やリソースを活用する代わりに、サブスクリプション等による支払いが必要な場合があります。これにより、費用を最適化し、簡単にスケールアップできます。サービスの例としては、使用量に応じて料金が発生する [Azure OpenAI Service](https://learn.microsoft.com/azure/ai-services/openai/overview?WT.mc_id=academic-105485-yoterada) があります。また、Azure OpenAI サービスは、モデルの能力に加え、エンタープライズグレードのセキュリティと責任ある AI のフレームワークも提供しています。 モデルは、パラメーターや重みなどを含むニューラルネットワークそのものです。企業はローカルでの運用も可能ですが、そのためには機器の購入、スケールアップのための環境構築、ライセンスの購入、またはオープンソース・モデルの使用が必要になります。LLaMA のようなモデルは利用可能ですが、モデルを実行するためには計算資源が必要です。 -## Azure でパフォーマンスを理解するために異なるモデルでテストと反復処理を行う方法 +## Azure でパフォーマンスを理解するために異なるモデルでテストと反復処理を行う方法 -現在の LLM の状況を調査し、シナリオに適したモデルの候補を見つけたら、次に企業の実データと負荷試験でそれらをテストする必要があります。これは実験と検証を通じて行われる反復的なプロセスです。前の段落で触れたモデル(OpenAIモデル、Llama2のようなオープンソースモデル、Hugging Faceのトランスフォーマー)のほとんどは、[Azure Machine Learning Studio](https://ml.azure.com/?WT.mc_id=academic-105485-yoterada) の[ファウンデーション・モデル・カタログ](https://learn.microsoft.com/azure/machine-learning/concept-foundation-models?WT.mc_id=academic-105485-yoterada)で利用可能です。 +現在の LLM の状況を調査し、シナリオに適したモデルの候補を見つけたら、次に企業の実データと負荷試験でそれらをテストする必要があります。これは実験と検証を通じて行われる反復的なプロセスです。前の段落で触れたモデル(OpenAI モデル、Llama2 のようなオープンソースモデル、Hugging Face のトランスフォーマー)のほとんどは、[Azure Machine Learning Studio](https://ml.azure.com/?WT.mc_id=academic-105485-yoterada) の[ファウンデーション・モデル・カタログ](https://learn.microsoft.com/azure/machine-learning/concept-foundation-models?WT.mc_id=academic-105485-yoterada)で利用可能です。 -[Azure Machine Learning](https://azure.microsoft.com/products/machine-learning/?WT.mc_id=academic-105485-yoterada) は、データ・サイエンティストと機械学習エンジニアが ML ライフサイクル全体(トレーニング、テスト、デプロイ、MLOpsの管理)を一つのプラットフォームで管理するために設計されたクラウド・サービスです。Machine Learning Studio はグラフィカル・ユーザー・インターフェースを提供し、利用者は下記の操作を行えます: +[Azure Machine Learning](https://azure.microsoft.com/products/machine-learning/?WT.mc_id=academic-105485-yoterada) は、データ・サイエンティストと機械学習エンジニアが ML ライフサイクル全体(トレーニング、テスト、デプロイ、MLOps の管理)を一つのプラットフォームで管理するために設計されたクラウド・サービスです。Machine Learning Studio はグラフィカル・ユーザー・インターフェースを提供し、利用者は下記の操作を行えます: - カタログから興味のあるファウンデーション・モデルを探し、タスク、ライセンス、名前でフィルタリングできます。カタログにまだ含まれていない新しいモデルもインポートできます。 - モデル・カードを確認して、詳細な説明とコードサンプルを見て、サンプル推論ウィジェットを使ってテストします。これは、サンプルのプロンプトを入力して回答結果を試すために利用します。 @@ -126,17 +126,17 @@ LLM モデルには、さまざまな種類があり、どのモデルを選択 ## 大規模言語モデル (LLM) の出力結果を改善する -スタートアップ・チームは、さまざまな種類の大規模言語モデル (LLM) とクラウド・プラットフォームのサービス(Azure Machine Learning)を理解し、異なるモデルを比較し、テストデータで評価し、パフォーマンスを向上させ、推論エンドポイントにデプロイする方法を検討しました。 +スタートアップ・チームは、さまざまな種類の大規模言語モデル (LLM) とクラウド・プラットフォームのサービス(Azure Machine Learning)を理解し、異なるモデルを比較し、テストデータで評価し、パフォーマンスを向上させ、推論エンドポイントにデプロイする方法を検討しました。 -しかし、事前トレーニングされたモデルを使用するのではなく、モデルをファイン・チューニングするのを検討すべきタイミングはいつでしょうか?特定の負荷試験でモデルのパフォーマンスを向上させる他のアプローチはあるのでしょうか? +しかし、事前トレーニングされたモデルを使用するのではなく、モデルをファイン・チューニングするのを検討すべきタイミングはいつでしょうか?特定の負荷試験でモデルのパフォーマンスを向上させる他のアプローチはあるのでしょうか? 企業が大規模言語モデル(LLM)を利用して、期待する結果を得るためには、トレーニング・レベルの異なる様々な手法からも選択できます。 -異なる実装方法(容易→困難)、コスト(低額→高額)、品質(低→高) から実装方法を検討し、LLM を本番環境にデプロイできます。以下に、いくつかの異なるアプローチを紹介します。 +異なる実装方法(容易 → 困難)、コスト(低額 → 高額)、品質(低 → 高) から実装方法を検討し、LLM を本番環境にデプロイできます。以下に、いくつかの異なるアプローチを紹介します。 - **コンテキストを用いたプロンプトエンジニアリング**:プロンプトを記述する際に十分なコンテキストを提供し、必要な回答を得るのが狙いです。 -- **Retrieval Augmented Generation(RAG)**:例えば、データがデータ・ベースや Web 上に存在する場合、関連データを取得し、プロンプトの記述時に、それらデータの一部をプロンプトに含めます。 +- **Retrieval Augmented Generation(RAG)**:例えば、データがデータ・ベースや Web 上に存在する場合、関連データを取得し、プロンプトの記述時に、それらデータの一部をプロンプトに含めます。 - **ファイン・チューニングしたモデル**:自分のデータを利用してモデルをさらにトレーニングし、モデルをより正確に、そしてニーズに応じた形にします。ただし、コストがかかる可能性があります。 @@ -144,7 +144,7 @@ LLM モデルには、さまざまな種類があり、どのモデルを選択 画像出展:: [Four Ways that Enterprises Deploy LLMs | Fiddler AI Blog](https://www.fiddler.ai/blog/four-ways-that-enterprises-deploy-llms?WT.mc_id=academic-105485-yoterada) -### コンテキストを用いたプロンプトエンジニアリング +### コンテキストを用いたプロンプトエンジニアリング 事前学習済みの大規模言語モデル(LLM)は、一般的な自然言語タスクとして、例えば、文の作成や質問などの処理をリクエストする際、短いプロンプトの呼び出しだけで、とても良い結果を出力します。これを「ゼロショット」学習と呼びます。 @@ -160,27 +160,27 @@ RAG は Vector データベース([Azure AI Search](https://learn.microsoft.co この技術は、LLM のファイン・チューニングが困難(十分なデータ、時間、リソースがない)な企業にとって非常に有用です。 そして特定の負荷試験におけるパフォーマンスを向上させたい場合や、回答の捏造リスク、つまり現実の歪曲や、有害なコンテンツのリスクを減らしたい場合に非常に有効です。 -### ファイン・チューニングしたモデルの利用 +### ファイン・チューニングしたモデルの利用 ファイン・チューニングは、転移学習を活用してモデルを下流タスクに「適応」させたり、特定の問題を解決するプロセスです。上記のフュー・ショット学習や、RAG とは異なり、更新した重みとバイアスを持つ新しいモデルを生成します。これには、入力(プロンプト)とそれに関連する出力(Completion)から構成する大量のトレーニング・データが必要です。このアプローチは、下記のような場合に有効です: -- **ファイン・チューニングしたモデルを使用する場合**:企業が高性能なモデルではなく、ファイン・チューニングした能力の低いモデル(埋め込みモデルなど)を使用し、よりコスト効率を高く、迅速なソリューション提供したいと考えている場合 +- **ファイン・チューニングしたモデルを使用する場合**:企業が高性能なモデルではなく、ファイン・チューニングした能力の低いモデル(埋め込みモデルなど)を使用し、よりコスト効率を高く、迅速なソリューション提供したいと考えている場合 -- **レイテンシーを考慮する場合**:特定の用途でレイテンシーが重要で、とても長いプロンプトを使用できない、またはモデルから学習するサンプル数がプロンプトの長さ制限に合わないような場合 +- **レイテンシーを考慮する場合**:特定の用途でレイテンシーが重要で、とても長いプロンプトを使用できない、またはモデルから学習するサンプル数がプロンプトの長さ制限に合わないような場合 - **最新の状態を維持する場合**:企業が高品質のデータと正確なラベルを多く持ち、これらのデータを時間をかけて最新の状態に保つためのリソースを持っている場合 -### トレーニング済みモデル +### トレーニング済みモデル LLM をゼロからトレーニングするのは、間違いなく最も困難で最も複雑なアプローチです。膨大なデータ、熟練したリソース、適切な計算能力が必要です。このオプションは、ビジネスがドメイン固有のユースケースと大量のドメイン中心のデータを持っている場合にのみ検討すべきです。 -## 知識チェック +## 知識チェック -LLM からの出力結果を改善するための良いアプローチは何でしょうか? +LLM からの出力結果を改善するための良いアプローチは何でしょうか? -1. コンテキストを用いたプロンプトエンジニアリング -2. RAG -3. ファイン・チューニングしたモデル +1. コンテキストを用いたプロンプトエンジニアリング +2. RAG +3. ファイン・チューニングしたモデル A: 3、時間とリソース、高品質のデータがある場合 ファイン・チューニングは最新の状態を維持するためのより良い選択肢です。しかし、改善を行うために時間がない場合は、まず RAG をご検討ください。 @@ -190,10 +190,10 @@ A: 3、時間とリソース、高品質のデータがある場合 ファイン ## 🚀 Challenge -ビジネスで RAG を活用する方法についてもっと学びたい方は、[こちら](https://learn.microsoft.com/azure/search/retrieval-augmented-generation-overview?WT.mc_id=academic-105485-yoterada)をご覧ください。 +ビジネスで RAG を活用する方法についてもっと学びたい方は、[こちら](https://learn.microsoft.com/azure/search/retrieval-augmented-generation-overview?WT.mc_id=academic-105485-yoterada)をご覧ください。 ## お疲れ様でした! 次のレッスンを続ける -このレッスン終了後、[生成 AI 学習コレクション](https://aka.ms/genai-collection?WT.mc_id=academic-105485-yoterada)をチェックして、生成 AI の知識をさらに深めましょう。 +このレッスン終了後、[生成 AI 学習コレクション](https://aka.ms/genai-collection?WT.mc_id=academic-105485-yoterada)をチェックして、生成 AI の知識をさらに深めましょう。 次のレッスン 3 では、[責任ある生成 AI の利用](../../../03-using-generative-ai-responsibly/translations/ja-jp/README.md?WT.mc_id=academic-105485-yoterada)について学びます! diff --git a/02-exploring-and-comparing-different-llms/translations/pt-br/README.md b/02-exploring-and-comparing-different-llms/translations/pt-br/README.md index 731a3298a..3de992254 100644 --- a/02-exploring-and-comparing-different-llms/translations/pt-br/README.md +++ b/02-exploring-and-comparing-different-llms/translations/pt-br/README.md @@ -1,8 +1,8 @@ # Explorando e comparando diferentes LLMs -[![Exploring and comparing different LLMs](../../images/02-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/J1mWzw0P74c?WT.mc_id=academic-105485-koreyst) +[![Exploring and comparing different LLMs](../../images/02-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=36c6795a-e63c-46dd-8d69-df8bbe6e7bc9?WT.mc_id=academic-105485-koreyst) -> *Clique na imagem acima para ver o vídeo desta lição* +> _Clique na imagem acima para ver o vídeo desta lição_ Na lição anterior, vimos como a IA generativa está mudando o cenário tecnológico, como os Grandes Modelos de Linguagens (LLMs) funcionam e como uma empresa - como nossa startup - pode aplicá-los aos seus casos de uso e crescer! Neste capítulo, estamos procurando comparar e contrastar diferentes tipos de modelos de linguagem grandes, LLMs para entender seus prós e contras. @@ -197,4 +197,3 @@ Saiba mais sobre como você pode [usar o RAG](https://learn.microsoft.com/azure/ Deseja aprender mais sobre diferentes conceitos de IA Generativa? Acesse a [página de aprendizado contínuo](../../../13-continued-learning/translations/pt-br/README.md?WT.mc_id=academic-105485-koreyst) para encontrar outros ótimos recursos sobre este tópico. Vamos para a Lição 3, onde veremos como podemos [Criar IA Generativa de forma Responsável](../../../03-using-generative-ai-responsibly/translations/pt-br/README.md?WT.mc_id=academic-105485-koreyst)! - diff --git a/04-prompt-engineering-fundamentals/README.md b/04-prompt-engineering-fundamentals/README.md index 329786fce..5021956dc 100644 --- a/04-prompt-engineering-fundamentals/README.md +++ b/04-prompt-engineering-fundamentals/README.md @@ -1,6 +1,6 @@ # Prompt Engineering Fundamentals -[![Prompt Engineering Fundamentals](./images/04-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/r2ItK3UMVTk?WT.mc_id=academic-105485-koreyst) +[![Prompt Engineering Fundamentals](./images/04-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=d54c0c69-b183-4a6c-80ed-8b1a8f299cff?WT.mc_id=academic-105485-koreyst) The way your write your prompt to an LLM also matters. A carefully-crafted prompt can achieve a better quality of response. But what exactly do terms like _prompt_ and _prompt engineering_ mean? And how do I improve the prompt _input_ that I send to the LLM? These are the questions we'll try to answer with in this chapter and the next. @@ -29,7 +29,7 @@ The Jupyter Notebook accompanying this lesson provides a _sandbox_ environment w 1. **An Azure OpenAI API key** - the service endpoint for a deployed LLM. 2. **A Python Runtime** - in which the Notebook can be executed. -3. **Local Env Variables** - *complete the [SETUP](SETUP.md) steps now to get ready*. +3. **Local Env Variables** - _complete the [SETUP](SETUP.md) steps now to get ready_. The notebook comes with _starter_ exercises - but you are encouraged to add your own _Markdown_ (description) and _Code_ (prompt requests) sections to try out more examples or ideas - and build your intuition for prompt design. diff --git a/04-prompt-engineering-fundamentals/translations/cn/README.md b/04-prompt-engineering-fundamentals/translations/cn/README.md index 6cffc0be3..2fcb5ab01 100644 --- a/04-prompt-engineering-fundamentals/translations/cn/README.md +++ b/04-prompt-engineering-fundamentals/translations/cn/README.md @@ -1,15 +1,14 @@ # 第四章:提示工程基础 -[![Prompt Engineering Fundamentals](../../images/04-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/r2ItK3UMVTk?WT.mc_id=academic-105485-koreyst) - +[![Prompt Engineering Fundamentals](../../images/04-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=d54c0c69-b183-4a6c-80ed-8b1a8f299cff?WT.mc_id=academic-105485-koreyst) 如何撰写 LLM 的提示很重要,精心设计的提示可以比不精心设计的提示取得更好的结果。 但这些概念到底是什么,提示、提示工程以及我如何改进我发送给 LLMs 的内容? 诸如此类的问题正是本章和下一章想要解答的。 -_生成式人工智能_能够根据用户请求创建新内容(例如文本、图像、音频、代码等)。 它使用 LLMs 来实现这一目标,例如 OpenAI 的 GPT 模型系列,这些模型通过使用自然语言和代码进行训练。 +*生成式人工智能*能够根据用户请求创建新内容(例如文本、图像、音频、代码等)。 它使用 LLMs 来实现这一目标,例如 OpenAI 的 GPT 模型系列,这些模型通过使用自然语言和代码进行训练。 用户现在可以使用熟悉的语言(如聊天)与这些模型进行交互,而无需任何技术专业知识或培训。 这些模型是基于提示的——用户发送文本输入(提示)并获取人工智能响应(完成)。 然后,他们可以在多轮对话中迭代地“与人工智能聊天”,完善他们的提示,直到响应符合他们的预期。 -“提示”现在成为生成式人工智能应用程序的主要_编程界面_,告诉模型要做什么并影响返回响应的质量。 “提示工程”是一个快速发展的研究领域,专注于提示的“设计和优化”,以大规模提供一致且高质量的响应。 +“提示”现在成为生成式人工智能应用程序的主要*编程界面*,告诉模型要做什么并影响返回响应的质量。 “提示工程”是一个快速发展的研究领域,专注于提示的“设计和优化”,以大规模提供一致且高质量的响应。 ## 学习目标 @@ -26,26 +25,24 @@ _生成式人工智能_能够根据用户请求创建新内容(例如文本、 目前,提示工程更多的是玄学而不是科学。 提高我们直觉的最佳方法是“更多练习”并采用试错方法,将应用程序领域的专业知识与推荐的技术和特定于模型的优化相结合。 -本课程附带的 Jupyter Notebook 提供了一个_沙盒_环境,您可以在其中尝试所学内容 - 边学边做,或者作为最后代码挑战的一部分。 要完成练习,您需要: +本课程附带的 Jupyter Notebook 提供了一个*沙盒*环境,您可以在其中尝试所学内容 - 边学边做,或者作为最后代码挑战的一部分。 要完成练习,您需要: 1. 设置 OpenAI API 密钥 - 已部署的 LLM 的服务端点。 2. Python 运行时 - 可以让 Notebook 运行。 -我们使用一个带有 Python 3 运行时的开发容器来检测这个存储库。 只需在 GitHub Codespaces 或本地 Docker 桌面上打开 Repo,即可自动激活运行时。 然后打开笔记本并选择Python 3.x 内核以准备 Notebook。 +我们使用一个带有 Python 3 运行时的开发容器来检测这个存储库。 只需在 GitHub Codespaces 或本地 Docker 桌面上打开 Repo,即可自动激活运行时。 然后打开笔记本并选择 Python 3.x 内核以准备 Notebook。 默认 Notebook 设置为与 OpenAI API 密钥一起使用。 只需将文件夹根目录中的“.env.copy”文件复制到“.env”,并使用您的 API 密钥更新“OPENAI_API_KEY=”行 - 一切就完成了。 -该 Notebook 附带入门练习 - 但我们鼓励您添加自己的Markdown(描述)和代码(提示请求)部分来尝试更多示例或想法 - 并建立您对提示工程设计的感觉。 - +该 Notebook 附带入门练习 - 但我们鼓励您添加自己的 Markdown(描述)和代码(提示请求)部分来尝试更多示例或想法 - 并建立您对提示工程设计的感觉。 ## Our Startup 的使命 - 现在,让我们来谈谈这个主题与 Our Startup 的使命[将人工智能创新带入教育](https://educationblog.microsoft.com/2023/06/collaborating-to-bring-ai-innovation-to-education?WT.mc_id=academic-105485-koreyst)有何关系 。 我们希望构建由人工智能驱动的个性化学习应用程序 - 所以让我们考虑一下我们应用程序的针对不同用户如何“设计”提示: -- **管理员**可能会要求人工智能分析课程数据以识别覆盖范围的差距_。 人工智能可以总结结果或用代码将其可视化。 -- **教育者**可能会要求人工智能为目标受众和主题生成教学计划。 AI可以按照指定的格式构建个性化计划。 +- **管理员**可能会要求人工智能分析课程数据以识别覆盖范围的差距\_。 人工智能可以总结结果或用代码将其可视化。 +- **教育者**可能会要求人工智能为目标受众和主题生成教学计划。 AI 可以按照指定的格式构建个性化计划。 - **学生**可能会要求人工智能辅导他们学习困难的科目。 人工智能现在可以通过适合学生水平的课程、结合提示和示例来指导学生。 这只是冰山一角。 查看 [教育中的提示工程](https://github.com/microsoft/prompts-for-edu/tree/main?WT.mc_id=academic-105485-koreyst) - 一个由教育专家设计的开源提示库 ! 尝试在沙箱中运行其中一些提示或使用 OpenAI Playground 看看会产生什么结果! @@ -91,7 +88,7 @@ LLM 将提示视为标记序列,其中不同的模型(或模型的版本) ![Base LLM Chat Completion](../../images/04-playground-chat-base.png?WT.mc_id=academic-105485-koreyst) -### 概念: LLMs 中的指令调整 +### 概念: LLMs 中的指令调整 [ LLMs 中的指令调整](https://blog.gopenai.com/an-introduction-to-base-and-instruction-tuned-large-language-models-8de102c785a6?WT.mc_id=academic-105485-koreyst) 从基础模型开始,并使用以下参数对其进行微调 可以包含明确指令的示例或输入/输出对(例如多轮“消息”),以及人工智能尝试遵循该指令的响应。 @@ -183,9 +180,8 @@ Illustrated with examples. 让我们从基础提示开始:发送到模型的文本输入,没有其他上下文。 这是一个例子 - 当我们将美国国歌的前几个单词发送到 OpenAI [Completion API](https://platform.openai.com/docs/api-reference/completions?WT.mc_id=academic-105485-koreyst) 时,它会立即完成下一个单词的响应 几行,说明了基本的预测行为。 - -| Prompt (Input) | Completion (Output) | -|:---|:---| +| Prompt (Input) | Completion (Output) | +| :----------------- | :----------------------------------------------------------------------------------------------------------------------------------------- | | Oh say can you see | It sounds like you're starting the lyrics to "The Star-Spangled Banner," the national anthem of the United States. The full lyrics are ... | ### 复杂的提示 @@ -197,7 +193,6 @@ Illustrated with examples. 该请求现在采用以下形式,其中标记化有效地从上下文和对话中捕获相关信息。 现在,更改系统上下文对补全质量的影响并与提供的用户输入一样。 - ```python response = openai.ChatCompletion.create( model="gpt-3.5-turbo", @@ -214,11 +209,11 @@ response = openai.ChatCompletion.create( 在上面的示例中,用户提示是一个简单的文本查询,可以解释为信息请求。 通过指令提示,我们可以使用该文本更详细地指定任务,从而为人工智能提供更好的指导。 这是一个例子: -| Prompt (Input) | Completion (Output) | Instruction Type | -|:---|:---|:---| -| Write a description of the Civil War | _returned a simple paragraph_ | Simple| -|Write a description of the Civil War. Provide key dates and events and describe their significance | _returned a paragraph followed by a list of key event dates with descriptions_| Complex | -|Write a description of the Civil War in 1 paragraph. Provide 3 bullet points with key dates and their significance. Provide 3 more bullet points with key historical figures and their contributions. Return the output as a JSON file | _returns more extensive details in a text box, formatted as JSON that you can copy-paste to a file and validate as needed_| Complex. Formatted. | +| Prompt (Input) | Completion (Output) | Instruction Type | +| :------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | :------------------------------------------------------------------------------------------------------------------------- | :------------------ | +| Write a description of the Civil War | _returned a simple paragraph_ | Simple | +| Write a description of the Civil War. Provide key dates and events and describe their significance | _returned a paragraph followed by a list of key event dates with descriptions_ | Complex | +| Write a description of the Civil War in 1 paragraph. Provide 3 bullet points with key dates and their significance. Provide 3 more bullet points with key historical figures and their contributions. Return the output as a JSON file | _returns more extensive details in a text box, formatted as JSON that you can copy-paste to a file and validate as needed_ | Complex. Formatted. | ## 主要内容 @@ -229,8 +224,8 @@ response = openai.ChatCompletion.create( 下面是一个示例,其中的指令是“用 2 句话总结这一点”。 -| Prompt (Input) | Completion (Output) | -|:---|:---| +| Prompt (Input) | Completion (Output) | +| :-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | :---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | | Jupiter is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets in the Solar System combined. Jupiter is one of the brightest objects visible to the naked eye in the night sky, and has been known to ancient civilizations since before recorded history. It is named after the Roman god Jupiter.[19] When viewed from Earth, Jupiter can be bright enough for its reflected light to cast visible shadows,[20] and is on average the third-brightest natural object in the night sky after the Moon and Venus.
**Summarize this in 2 short sentences** | Jupiter, the fifth planet from the Sun, is the largest in the Solar System and is known for being one of the brightest objects in the night sky. Named after the Roman god Jupiter, it's a gas giant whose mass is two-and-a-half times that of all other planets in the Solar System combined. | 主要内容片段可以通过多种方式使用来驱动更有效的指令: @@ -239,7 +234,6 @@ response = openai.ChatCompletion.create( - **暗示** - 遵循带有“暗示”的说明,引导完成,引导模型做出更相关的响应。 - **模板** - 这些是带有占位符(变量)提示的可重复“配方”,可以使用特定用例的数据进行自定义。 - ### 用例方式 这种方法中,您使用主要内容来“向模型提供”给定指令所需输出的一些示例,并让它推断所需输出的模式。 根据提供的示例数量,我们可以有零样本提示、单样本提示、少样本提示等。 @@ -250,14 +244,12 @@ response = openai.ChatCompletion.create( - 所需输出的一些示例 - 新示例的开始(成为隐式任务描述) - -| Learning Type | Prompt (Input) | Completion (Output) | -|:---|:---|:---| -| Zero-shot | "The Sun is Shining". Translate to Spanish | "El Sol está brillando".| -| One-shot | "The Sun is Shining" => ""El Sol está brillando".
"It's a Cold and Windy Day" => | "Es un día frío y ventoso". | -| Few-shot | The player ran the bases => Baseball
The player hit an ace => Tennis
The player hit a six => Cricket
The player made a slam-dunk => | Basketball | -| | | | - +| Learning Type | Prompt (Input) | Completion (Output) | +| :------------ | :---------------------------------------------------------------------------------------------------------------------------------------------------- | :-------------------------- | +| Zero-shot | "The Sun is Shining". Translate to Spanish | "El Sol está brillando". | +| One-shot | "The Sun is Shining" => ""El Sol está brillando".
"It's a Cold and Windy Day" => | "Es un día frío y ventoso". | +| Few-shot | The player ran the bases => Baseball
The player hit an ace => Tennis
The player hit a six => Cricket
The player made a slam-dunk => | Basketball | +| | | | 请注意,我们如何必须在零样本提示中提供明确的指令(“翻译为西班牙语”),但它是在一次提示示例中推断出来的。 这个少样本示例展示了如何添加更多示例来让模型在不添加指令的情况下做出更准确的推理。 @@ -265,26 +257,24 @@ response = openai.ChatCompletion.create( 使用主要内容的另一种技术是提供暗示而不是示例。 在这种情况下,我们通过使用反映所需响应格式的片段启动来让模型朝正确的方向进行推理。 然后,模型“接受提示”,继续按照这种思路进行。 - -| Number of Cues | Prompt (Input) | Completion (Output) | -|:---|:---|:---| -| 0 | Jupiter is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets in the Solar System combined. Jupiter is one of the brightest objects visible to the naked eye in the night sky, and has been known to ancient civilizations since before recorded history.
**Summarize This** | Jupiter is the largest planet in our Solar System and the fifth one from the Sun. It is a gas giant with a mass 1/1000th of the Sun's, but it is heavier than all the other planets combined. Ancient civilizations have known about Jupiter for a long time, and it is easily visible in the night sky..| -| 1 | Jupiter is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets in the Solar System combined. Jupiter is one of the brightest objects visible to the naked eye in the night sky, and has been known to ancient civilizations since before recorded history.
**Summarize This**
What we learned is that Jupiter | is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets combined. It is easily visible to the naked eye and has been known since ancient times. | -| 2 | Jupiter is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets in the Solar System combined. Jupiter is one of the brightest objects visible to the naked eye in the night sky, and has been known to ancient civilizations since before recorded history.
**Summarize This**
Top 3 Facts We Learned: | 1. Jupiter is the fifth planet from the Sun and the largest in the Solar System.
2. It is a gas giant with a mass one-thousandth that of the Sun...
3. Jupiter has been visible to the naked eye since ancient times ... | -| | | | - +| Number of Cues | Prompt (Input) | Completion (Output) | +| :------------- | :----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | :-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| 0 | Jupiter is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets in the Solar System combined. Jupiter is one of the brightest objects visible to the naked eye in the night sky, and has been known to ancient civilizations since before recorded history.
**Summarize This** | Jupiter is the largest planet in our Solar System and the fifth one from the Sun. It is a gas giant with a mass 1/1000th of the Sun's, but it is heavier than all the other planets combined. Ancient civilizations have known about Jupiter for a long time, and it is easily visible in the night sky.. | +| 1 | Jupiter is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets in the Solar System combined. Jupiter is one of the brightest objects visible to the naked eye in the night sky, and has been known to ancient civilizations since before recorded history.
**Summarize This**
What we learned is that Jupiter | is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets combined. It is easily visible to the naked eye and has been known since ancient times. | +| 2 | Jupiter is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets in the Solar System combined. Jupiter is one of the brightest objects visible to the naked eye in the night sky, and has been known to ancient civilizations since before recorded history.
**Summarize This**
Top 3 Facts We Learned: | 1. Jupiter is the fifth planet from the Sun and the largest in the Solar System.
2. It is a gas giant with a mass one-thousandth that of the Sun...
3. Jupiter has been visible to the naked eye since ancient times ... | +| | | | ### 提示模版 -提示模板是预定义的提示配方,可以根据需要进行存储和重用,以大规模推动更一致的用户体验。 最简单的形式是,它只是一组提示示例的集合,例如 [OpenAI 中的这个例子](https://platform.openai.com/examples?WT.mc_id=academic-105485-koreyst),它提供了交互式提示组件(用户和系统消息)和 AP驱动请求格式来支持重用。 +提示模板是预定义的提示配方,可以根据需要进行存储和重用,以大规模推动更一致的用户体验。 最简单的形式是,它只是一组提示示例的集合,例如 [OpenAI 中的这个例子](https://platform.openai.com/examples?WT.mc_id=academic-105485-koreyst),它提供了交互式提示组件(用户和系统消息)和 AP 驱动请求格式来支持重用。 -在它更复杂的形式中,比如[LangChain的这个例子](https://python.langchain.com/docs/modules/model_io/prompts/prompt_templates/?WT.mc_id=academic-105485-koreyst),它包含占位符,可以替换为来自各种来源的数据(用户 输入、系统上下文、外部数据源等)来动态生成提示。 这使我们能够创建一个可重用的提示库,可用于大规模地**以编程方式**驱动一致的用户体验。 +在它更复杂的形式中,比如[LangChain 的这个例子](https://python.langchain.com/docs/modules/model_io/prompts/prompt_templates/?WT.mc_id=academic-105485-koreyst),它包含占位符,可以替换为来自各种来源的数据(用户 输入、系统上下文、外部数据源等)来动态生成提示。 这使我们能够创建一个可重用的提示库,可用于大规模地**以编程方式**驱动一致的用户体验。 最后,模板的真正价值在于能够为垂直应用程序领域创建和发布提示库 - 其中提示模板现在已优化以反映特定于应用程序的上下文或示例,使响应对于目标用户受众更加相关和准确 。 [Prompts For Edu](https://github.com/microsoft/prompts-for-edu?WT.mc_id=academic-105485-koreyst) repo 是这种方法的一个很好的例子,它为教育领域策划了一个提示库,重点关注课程计划等关键目标, 课程设计、学生辅导等 ## 支持内容 -如果我们将提示构建视为具有指令(任务) 和目标(主要内容),那么次要内容_像我们提供的附加上下文**以某种方式影响输出**。 它可以是调整参数、格式化指令、主题分类法等,可以帮助模型定制其响应以适应所需的用户目标或期望。 +如果我们将提示构建视为具有指令(任务) 和目标(主要内容),那么次要内容\_像我们提供的附加上下文**以某种方式影响输出**。 它可以是调整参数、格式化指令、主题分类法等,可以帮助模型定制其响应以适应所需的用户目标或期望。 例如:给定一个包含课程表中所有可用课程的广泛元数据(名称、描述、级别、元数据标签、讲师等)的课程目录: @@ -294,7 +284,6 @@ response = openai.ChatCompletion.create( 现在,该模型可以按照几个示例所示的格式提供摘要 - 但如果结果具有多个标签,它可以优先考虑辅助内容中标识的 5 个标签。 - --- - ## 知识检查 以下哪项是遵循一些合理的最佳实践的最佳提示? 1. Show me an image of red car 2. Show me an image of red car of make Volvo and model XC90 parked by a cliff with the sun setting -3. Show me an image of red car of make Volvo and model XC90 +3. Show me an image of red car of make Volvo and model XC90 答:2,这是最好的提示,因为它提供了有关“内容”的详细信息并详细说明(不仅仅是任何汽车,而是特定的品牌和型号),并且还描述了整体设置。 3 是次佳的,因为它也包含很多描述。 diff --git a/04-prompt-engineering-fundamentals/translations/ja-jp/README.md b/04-prompt-engineering-fundamentals/translations/ja-jp/README.md index 76c700ce8..7b996559a 100644 --- a/04-prompt-engineering-fundamentals/translations/ja-jp/README.md +++ b/04-prompt-engineering-fundamentals/translations/ja-jp/README.md @@ -1,8 +1,8 @@ # プロンプト・エンジニアリングの基礎 -[![Prompt Engineering Fundamentals](../../images/04-lesson-banner.png?WT.mc_id=academic-105485-yoterada)](https://youtu.be/r2ItK3UMVTk?WT.mc_id=academic-105485-yoterada) +[![Prompt Engineering Fundamentals](../../images/04-lesson-banner.png?WT.mc_id=academic-105485-yoterada)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=d54c0c69-b183-4a6c-80ed-8b1a8f299cff?WT.mc_id=academic-105485-yoterada) -大規模言語モデル (LLM) では、プロンプトの書き方がとても重要で、慎重に作成したプロンプトは、そうでないものに比べ良い結果をもたらします。しかし、プロンプトやプロンプト・エンジニアリングとは一体どういう物なのでしょうか?また、LLM に送信する内容をどのようにして改善すればいいのでしょうか?この章と次の章では、そうした疑問に答えたいと思います。 +大規模言語モデル (LLM) では、プロンプトの書き方がとても重要で、慎重に作成したプロンプトは、そうでないものに比べ良い結果をもたらします。しかし、プロンプトやプロンプト・エンジニアリングとは一体どういう物なのでしょうか?また、LLM に送信する内容をどのようにして改善すればいいのでしょうか?この章と次の章では、そうした疑問に答えたいと思います。 「_生成 AI_」は、利用者からのリクエストに対して、テキスト、画像、オーディオ、コード等の新しいコンテンツを生成する能力を持っています。これを実現するために、OpenAI の GPT(Generative Pre-trained Transformer)シリーズのような、自然言語とコードの使用目的でトレーニングされた「_大規模言語モデル(LLM)_」を利用できます。 @@ -38,13 +38,13 @@ ## スタートアップ -それでは、_このトピック_ がスタートアップのミッションである「[教育へ AI 革新をもたらす](https://educationblog.microsoft.com/2023/06/collaborating-to-bring-ai-innovation-to-education?WT.mc_id=academic-105485-yoterada)」とどのように結びついているのかについて見て行きましょう。私たちは「_個別学習_」を実現する AI アプリケーションの開発を目指しています。そこで、私たちのアプリケーションを利用する様々な利用者がプロンプトをどのように「デザイン」するのかを考えてみましょう。 +それでは、_このトピック_ がスタートアップのミッションである「[教育へ AI 革新をもたらす](https://educationblog.microsoft.com/2023/06/collaborating-to-bring-ai-innovation-to-education?WT.mc_id=academic-105485-yoterada)」とどのように結びついているのかについて見て行きましょう。私たちは「_個別学習_」を実現する AI アプリケーションの開発を目指しています。そこで、私たちのアプリケーションを利用する様々な利用者がプロンプトをどのように「デザイン」するのかを考えてみましょう。 - **管理者**: AI に「_カリキュラムのデータを分析し、カバーしてない領域を特定してください_」と指示する可能性があります。AI はその結果をまとめたり、コードを用いて可視化できます。 - **教員**: AI に「_特定の生徒と授業科目に応じた授業計画を作成してください_ 」と指示する可能性があります。AI は指定されたフォーマットに従って個別に計画を作成できます。 - **生徒**: AI に「_苦手な科目に対して個別指導をしてください_」と指示する可能性があります。AI は生徒のレベルに合わせたレッスン、ヒント、例を提供して指導できます。 -これらは、ほんの一例です。教育専門家から厳選された、オープンソースのプロンプト・ライブラリ「[Prompts For Education](https://github.com/microsoft/prompts-for-edu/tree/main?WT.mc_id=academic-105485-yoterada)」を確認し、さらに広い視野で可能性を探ってみてください! _サンドボックスでそれらのプロンプトを試しに実行したり、OpenAI Playground で試してどのような結果が得られるかを試してください!_ +これらは、ほんの一例です。教育専門家から厳選された、オープンソースのプロンプト・ライブラリ「[Prompts For Education](https://github.com/microsoft/prompts-for-edu/tree/main?WT.mc_id=academic-105485-yoterada)」を確認し、さらに広い視野で可能性を探ってみてください! _サンドボックスでそれらのプロンプトを試しに実行したり、OpenAI Playground で試してどのような結果が得られるかを試してください!_ -## 課題 +## 課題 おめでとうございます!レッスンの最後までたどり着きました!これから、実際の例を使って、いくつかの概念や技術を試す時が来ました! 課題では、対話式で取り組める演習問題がセットされた Jupyter Notebook を使用します。また、ご自身でアイデアや技術を検証するために、独自の Markdown やコードセルを Notebook に追加できます。 -### 作業を開始するには、リポジトリをフォークしてください - -- (推奨) GitHub Codespacesを起動 +### 作業を開始するには、リポジトリをフォークしてください + +- (推奨) GitHub Codespaces を起動 - (代替案 1) リポジトリをローカル・デバイスにクローンして、ローカルの Docker Desktop で実行 -- (代替案 2) 好みのノートブック実行環境でノートブックを開く +- (代替案 2) 好みのノートブック実行環境でノートブックを開く ### 次に、環境変数を設定してください - リポジトリのルートにある `.env.copy` ファイルを `.env` にコピーし、`OPENAI_API_KEY` の値を入力してください。API キーは [OpenAI Dashboard](https://beta.openai.com/account/api-keys?WT.mc_id=academic-105485-yoterada) から取得できます。 -### 次に、Jupyter Notebookを開いてください +### 次に、Jupyter Notebook を開いてください - 使用する実行環境のカーネルを選択してください。オプション 1 や 2 をご利用される場合は、.devcontainer で提供するコンテナ・イメージの、Python 3.10.x カーネルを選んでください。 @@ -400,10 +400,10 @@ LESSON TEMPLATE: 以下のプロンプトの中で、適切なベストプラクティスに沿ったものはどれですか? 1. 赤い車の写真を画像を表示してください -2. 夕日が沈む崖のそばに駐車した、赤いボルボの XC90 モデルの画像を表示してください +2. 夕日が沈む崖のそばに駐車した、赤いボルボの XC90 モデルの画像を表示してください 3. 赤いボルボの XC90 モデルの画像を表示してください -A: 2番が最良のプロンプトです。それは「何を」求めているのかの詳細を提供し、具体的な情報(任意の車ではなく、特定のメーカーとモデル)を含んでいる上、周囲の環境についても描写しています。次に良いのは3番で、これも詳細な記述が含まれています。 +A: 2 番が最良のプロンプトです。それは「何を」求めているのかの詳細を提供し、具体的な情報(任意の車ではなく、特定のメーカーとモデル)を含んでいる上、周囲の環境についても描写しています。次に良いのは 3 番で、これも詳細な記述が含まれています。 ## 🚀 チャレンジ @@ -414,4 +414,4 @@ A: 2番が最良のプロンプトです。それは「何を」求めている プロンプト・エンジニアリングに関して、さらに概念をもっと深く学びたい方は、この話題に関する他の素晴らしい情報を見つけるために[継続的学習のページ](../../../13-continued-learning/README.md?WT.mc_id=academic-105485-yoterada)をご参照ください。 -それでは、レッスン5に移り、[高度なプロンプト技術](../../../05-advanced-prompts/translations/ja-jp/README.md?WT.mc_id=academic-105485-yoterada)を見ていきましょう! +それでは、レッスン 5 に移り、[高度なプロンプト技術](../../../05-advanced-prompts/translations/ja-jp/README.md?WT.mc_id=academic-105485-yoterada)を見ていきましょう! diff --git a/04-prompt-engineering-fundamentals/translations/pt-br/README.md b/04-prompt-engineering-fundamentals/translations/pt-br/README.md index 0839b86bc..65518f68e 100644 --- a/04-prompt-engineering-fundamentals/translations/pt-br/README.md +++ b/04-prompt-engineering-fundamentals/translations/pt-br/README.md @@ -1,6 +1,6 @@ # Fundamentos de Engenharia de Prompt -[![Prompt Engineering Fundamentals](../../images/04-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/r2ItK3UMVTk?WT.mc_id=academic-105485-koreyst) +[![Prompt Engineering Fundamentals](../../images/04-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=d54c0c69-b183-4a6c-80ed-8b1a8f299cff?WT.mc_id=academic-105485-koreyst) A forma como você escreve seu prompt para o LLM importa. Um prompt cuidadosamente elaborado pode alcançar um resultado melhor do que um que não é. Mas o que são esses conceitos, prompt, Engenharia de Prompt e como posso melhorar o que envio para o LLM? Perguntas como essas são o que este capítulo e o próximo estão procurando responder. @@ -80,7 +80,7 @@ Para ter uma intuição de como a tokenização funciona, experimente ferramenta ### Conceito: Modelos Fundamentais -Uma vez que um prompt é tokenizado, a função principal do ["Base LLM"](https://blog.gopenai.com/an-introduction-to-base-and-instruction-tuned-large-language-models-8de102c785a6?WT.mc_id=academic-105485-koreyst) (ou modelo fundamental) é prever o token nessa sequência. Como os LLMs são treinados em conjuntos massivos de dados de texto, eles têm uma boa compreensão das relações estatísticas entre tokens e podem fazer essa previsão com alguma confiança. +Uma vez que um prompt é tokenizado, a função principal do ["Base LLM"](https://blog.gopenai.com/an-introduction-to-base-and-instruction-tuned-large-language-models-8de102c785a6?WT.mc_id=academic-105485-koreyst) (ou modelo fundamental) é prever o token nessa sequência. Como os LLMs são treinados em conjuntos massivos de dados de texto, eles têm uma boa compreensão das relações estatísticas entre tokens e podem fazer essa previsão com alguma confiança. > Observação: eles não compreendem o _significado_ das palavras no prompt ou token; eles apenas veem um padrão que podem "completar" com sua próxima previsão. Eles podem continuar prevendo a sequência até serem interrompidos pela intervenção do usuário ou alguma condição preestabelecida. @@ -122,6 +122,7 @@ Vamos ver isso em ação no OpenAI ou Azure OpenAI Playground: ### Exemplo de Alucinações Quer ter uma ideia de como as alucinações funcionam? Pense em um prompt que instrua a IA a gerar conteúdo para um tópico inexistente (para garantir que não seja encontrado no conjunto de dados de treinamento). Por exemplo - eu tentei este prompt: + > **Prompt:** generate a lesson plan on the Martian War of 2076. Uma busca na web mostrou que havia relatos fictícios (por exemplo, séries de televisão ou livros) sobre guerras marcianas - mas nenhuma em 2076. O bom senso também nos diz que 2076 está _no futuro_ e, portanto, não pode ser associado a um evento real. @@ -181,8 +182,8 @@ Vimos por que a Engenharia de Prompt é importante - agora vamos entender como o Vamos começar com o prompt básico: uma entrada de texto enviada ao modelo sem nenhum outro contexto. Aqui está um exemplo - quando enviamos as primeiras palavras do hino nacional dos EUA para a [API de Completions da OpenAI](https://platform.openai.com/docs/api-reference/completions?WT.mc_id=academic-105485-koreyst), ela instantaneamente _completa_ a resposta com as próximas linhas, ilustrando o comportamento básico de previsão. -| Prompt (Input) | Completion (Output) | -|:---|:---| +| Prompt (Input) | Completion (Output) | +| :----------------- | :----------------------------------------------------------------------------------------------------------------------------------------- | | Oh say can you see | It sounds like you're starting the lyrics to "The Star-Spangled Banner," the national anthem of the United States. The full lyrics are ... | ### Prompt Complexo @@ -210,11 +211,11 @@ response = openai.ChatCompletion.create( Nos exemplos acima, o prompt do usuário era uma simples consulta de texto que pode ser interpretada como uma solicitação de informações. Com prompts de _instrução_, podemos usar esse texto para especificar uma tarefa de maneira mais detalhada, fornecendo orientações melhores para a IA. Aqui está um exemplo: -| Prompt (Input) | Completion (Output) | Instruction Type | -|:---|:---|:---| -| Write a description of the Civil War | _returned a simple paragraph_ | Simple| -|Write a description of the Civil War. Provide key dates and events and describe their significance | _returned a paragraph followed by a list of key event dates with descriptions_| Complex | -|Write a description of the Civil War in 1 paragraph. Provide 3 bullet points with key dates and their significance. Provide 3 more bullet points with key historical figures and their contributions. Return the output as a JSON file | _returns more extensive details in a text box, formatted as JSON that you can copy-paste to a file and validate as needed_| Complex. Formatted. | +| Prompt (Input) | Completion (Output) | Instruction Type | +| :------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | :------------------------------------------------------------------------------------------------------------------------- | :------------------ | +| Write a description of the Civil War | _returned a simple paragraph_ | Simple | +| Write a description of the Civil War. Provide key dates and events and describe their significance | _returned a paragraph followed by a list of key event dates with descriptions_ | Complex | +| Write a description of the Civil War in 1 paragraph. Provide 3 bullet points with key dates and their significance. Provide 3 more bullet points with key historical figures and their contributions. Return the output as a JSON file | _returns more extensive details in a text box, formatted as JSON that you can copy-paste to a file and validate as needed_ | Complex. Formatted. | ## Conteúdo Primário @@ -225,9 +226,8 @@ Nos exemplos acima, o prompt ainda era bastante aberto, permitindo que o LLM dec Aqui está um exemplo em que a instrução é: "resuma isso em 2 frases". - -| Prompt (Input) | Completion (Output) | -|:---|:---| +| Prompt (Input) | Completion (Output) | +| :-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | :---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | | Jupiter is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets in the Solar System combined. Jupiter is one of the brightest objects visible to the naked eye in the night sky, and has been known to ancient civilizations since before recorded history. It is named after the Roman god Jupiter.[19] When viewed from Earth, Jupiter can be bright enough for its reflected light to cast visible shadows,[20] and is on average the third-brightest natural object in the night sky after the Moon and Venus.
**Summarize this in 2 short sentences** | Jupiter, the fifth planet from the Sun, is the largest in the Solar System and is known for being one of the brightest objects in the night sky. Named after the Roman god Jupiter, it's a gas giant whose mass is two-and-a-half times that of all other planets in the Solar System combined. | O segmento de conteúdo primário pode ser usado de várias maneiras para impulsionar instruções mais eficazes: @@ -248,14 +248,12 @@ O prompt agora consiste em três componentes: - Alguns exemplos da saída desejada - O início de um novo exemplo (que se torna uma descrição implícita da tarefa) - - | Learning Type | Prompt (Input) | Completion (Output) | - |:---|:---|:---| - | Zero-shot | "The Sun is Shining". Translate to Spanish | "El Sol está brillando".| - | One-shot | "The Sun is Shining" => ""El Sol está brillando".
"It's a Cold and Windy Day" => | "Es un día frío y ventoso". | - | Few-shot | The player ran the bases => Baseball
The player hit an ace => Tennis
The player hit a six => Cricket
The player made a slam-dunk => | Basketball | - | | | | - + | Learning Type | Prompt (Input) | Completion (Output) | + | :------------ | :---------------------------------------------------------------------------------------------------------------------------------------------------- | :-------------------------- | + | Zero-shot | "The Sun is Shining". Translate to Spanish | "El Sol está brillando". | + | One-shot | "The Sun is Shining" => ""El Sol está brillando".
"It's a Cold and Windy Day" => | "Es un día frío y ventoso". | + | Few-shot | The player ran the bases => Baseball
The player hit an ace => Tennis
The player hit a six => Cricket
The player made a slam-dunk => | Basketball | + | | | | Observe como tivemos que fornecer instruções explícitas ("Traduza para o espanhol") no prompting de `zero-shot`, mas isso é inferido no exemplo de `one-shot`. O exemplo `few-shot` mostra como adicionar mais exemplos permite que os modelos façam inferências mais precisas sem instruções adicionais. @@ -263,13 +261,12 @@ Observe como tivemos que fornecer instruções explícitas ("Traduza para o espa Outra técnica para usar conteúdo primário é fornecer _dicas_ em vez de exemplos. Neste caso, estamos dando ao modelo um empurrão na direção certa ao _iniciá-lo_ com um trecho que reflete o formato da resposta desejada. O modelo então "pega a dica" para continuar nessa linha. - -| Number of Cues | Prompt (Input) | Completion (Output) | -|:---|:---|:---| -| 0 | Jupiter is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets in the Solar System combined. Jupiter is one of the brightest objects visible to the naked eye in the night sky, and has been known to ancient civilizations since before recorded history.
**Summarize This** | Jupiter is the largest planet in our Solar System and the fifth one from the Sun. It is a gas giant with a mass 1/1000th of the Sun's, but it is heavier than all the other planets combined. Ancient civilizations have known about Jupiter for a long time, and it is easily visible in the night sky..| -| 1 | Jupiter is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets in the Solar System combined. Jupiter is one of the brightest objects visible to the naked eye in the night sky, and has been known to ancient civilizations since before recorded history.
**Summarize This**
What we learned is that Jupiter | is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets combined. It is easily visible to the naked eye and has been known since ancient times. | -| 2 | Jupiter is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets in the Solar System combined. Jupiter is one of the brightest objects visible to the naked eye in the night sky, and has been known to ancient civilizations since before recorded history.
**Summarize This**
Top 3 Facts We Learned: | 1. Jupiter is the fifth planet from the Sun and the largest in the Solar System.
2. It is a gas giant with a mass one-thousandth that of the Sun...
3. Jupiter has been visible to the naked eye since ancient times ... | -| | | | +| Number of Cues | Prompt (Input) | Completion (Output) | +| :------------- | :----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | :-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| 0 | Jupiter is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets in the Solar System combined. Jupiter is one of the brightest objects visible to the naked eye in the night sky, and has been known to ancient civilizations since before recorded history.
**Summarize This** | Jupiter is the largest planet in our Solar System and the fifth one from the Sun. It is a gas giant with a mass 1/1000th of the Sun's, but it is heavier than all the other planets combined. Ancient civilizations have known about Jupiter for a long time, and it is easily visible in the night sky.. | +| 1 | Jupiter is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets in the Solar System combined. Jupiter is one of the brightest objects visible to the naked eye in the night sky, and has been known to ancient civilizations since before recorded history.
**Summarize This**
What we learned is that Jupiter | is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets combined. It is easily visible to the naked eye and has been known since ancient times. | +| 2 | Jupiter is the fifth planet from the Sun and the largest in the Solar System. It is a gas giant with a mass one-thousandth that of the Sun, but two-and-a-half times that of all the other planets in the Solar System combined. Jupiter is one of the brightest objects visible to the naked eye in the night sky, and has been known to ancient civilizations since before recorded history.
**Summarize This**
Top 3 Facts We Learned: | 1. Jupiter is the fifth planet from the Sun and the largest in the Solar System.
2. It is a gas giant with a mass one-thousandth that of the Sun...
3. Jupiter has been visible to the naked eye since ancient times ... | +| | | | Observe como tivemos que fornecer uma instrução explícita ("Resuma Isso") no exemplo de zero dicas, mas isso é inferido no exemplo de uma dica. O exemplo de duas dicas mostra como adicionar mais dicas permite que os modelos façam inferências mais precisas sem instruções adicionais. @@ -326,18 +323,17 @@ A Engenharia de Prompt é um processo de tentativa e erro, então tenha em mente Agora, vamos dar uma olhada nas práticas recomendadas comuns pela [Open AI](https://help.openai.com/en/articles/6654000-best-practices-for-prompt-engineering-with-openai-api?WT.mc_id=academic-105485-koreyst) e pelos praticantes da [Azure OpenAI](https://learn.microsoft.com/azure/ai-services/openai/concepts/prompt-engineering#best-practices?WT.mc_id=academic-105485-koreyst). - -| What | Why | -|:---|:---| -| Evaluate the latest models. | New model generations are likely to have improved features and quality - but may also incur higher costs. Evaluate them for impact, then make migration decisions. | -| Separate instructions & context | Check if your model/provider defines _delimiters_ to distinguish instructions, primary and secondary content more clearly. This can help models assign weights more accurately to tokens. | -|Be specific and clear | Give more details about the desired context, outcome, length, format, style etc. This will improve both the quality and consistency of responses. Capture recipes in reusable templates. | -|Be descriptive, use examples |Models may respond better to a "show and tell" approach. Start with a `zero-shot` approach where you give it an instruction (but no examples) then try `few-shot` as a refinement, providing a few examples of the desired output. Use analogies.| -| Use cues to jumpstart completions | Nudge it towards a desired outcome by giving it some leading words or phrases that it can use as a starting point for the response.| -|Double Down | Sometimes you may need to repeat yourself to the model. Give instructions before and after your primary content, use an instruction and a cue, etc. Iterate & validate to see what works.| -| Order Matters | The order in which you present information to the model may impact the output, even in the learning examples, thanks to recency bias. Try different options to see what works best.| -|Give the model an “out” | Give the model a _fallback_ completion response it can provide if it cannot complete the task for any reason. This can reduce chances of models generating false or hallucinatory responses. | -| | | +| What | Why | +| :-------------------------------- | :------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ | +| Evaluate the latest models. | New model generations are likely to have improved features and quality - but may also incur higher costs. Evaluate them for impact, then make migration decisions. | +| Separate instructions & context | Check if your model/provider defines _delimiters_ to distinguish instructions, primary and secondary content more clearly. This can help models assign weights more accurately to tokens. | +| Be specific and clear | Give more details about the desired context, outcome, length, format, style etc. This will improve both the quality and consistency of responses. Capture recipes in reusable templates. | +| Be descriptive, use examples | Models may respond better to a "show and tell" approach. Start with a `zero-shot` approach where you give it an instruction (but no examples) then try `few-shot` as a refinement, providing a few examples of the desired output. Use analogies. | +| Use cues to jumpstart completions | Nudge it towards a desired outcome by giving it some leading words or phrases that it can use as a starting point for the response. | +| Double Down | Sometimes you may need to repeat yourself to the model. Give instructions before and after your primary content, use an instruction and a cue, etc. Iterate & validate to see what works. | +| Order Matters | The order in which you present information to the model may impact the output, even in the learning examples, thanks to recency bias. Try different options to see what works best. | +| Give the model an “out” | Give the model a _fallback_ completion response it can provide if it cannot complete the task for any reason. This can reduce chances of models generating false or hallucinatory responses. | +| | | Como em qualquer prática recomendada, lembre-se de que _seus resultados podem variar_ com base no modelo, na tarefa e no domínio. Use essas práticas como ponto de partida e itere para encontrar o que funciona melhor para você. Reavalie constantemente seu processo de engenharia de prompt à medida que novos modelos e ferramentas se tornam disponíveis, com foco na escalabilidade do processo e na qualidade das respostas. diff --git a/05-advanced-prompts/README.md b/05-advanced-prompts/README.md index 418c1e669..86c7dfa85 100644 --- a/05-advanced-prompts/README.md +++ b/05-advanced-prompts/README.md @@ -1,6 +1,6 @@ # Creating Advanced prompts -[![Creating Advanced Prompts](./images/05-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/32GBH6BTWZQ?WT.mc_id=academic-105485-koreyst) +[![Creating Advanced Prompts](./images/05-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=f3615e46-dbca-477c-832d-21eef95cd9f3?WT.mc_id=academic-105485-koreyst) Let's recap some learnings from the previous chapter: @@ -8,7 +8,7 @@ Let's recap some learnings from the previous chapter: There are also two steps to writing prompts, constructing the prompt, by providing relevant context and the second part is _optimization_, how to gradually improve the prompt. -At this point, we have some basic understanding of how to write prompts, but we need to go deeper. In this chapter, you will go from trying out various prompts to understanding why one prompt is better than another. You will learn how to construct prompts following some basic techniques that can be applied to any LLM. +At this point, we have some basic understanding of how to write prompts, but we need to go deeper. In this chapter, you will go from trying out various prompts to understanding why one prompt is better than another. You will learn how to construct prompts following some basic techniques that can be applied to any LLM. ## Introduction @@ -112,7 +112,7 @@ Here's how: Alice has 5 apples, throws 3 apples, gives 2 to Bob and Bob gives one back, how many apples does Alice have?" Answer: 1 -Note how we write substantially longer prompts with another example, a calculation and then the original prompt and we arrive at the correct answer 1. +Note how we write substantially longer prompts with another example, a calculation and then the original prompt and we arrive at the correct answer 1. As you can see chain-of-thought is a very powerful technique. @@ -124,7 +124,7 @@ As an example, this is what your prompt then can look like if you're in the insu ```text {{company}}: {{company_name}} - {{products}}: + {{products}}: {{products_list}} Please suggest an insurance given the following budget and requirements: Budget: {{budget}} @@ -137,13 +137,13 @@ Here's an example of how the prompt could look like once the variables have been ```text Insurance company: ACME Insurance - Insurance products (cost per month): + Insurance products (cost per month): - Car, cheap, 500 USD - - Car, expensive, 1100 USD + - Car, expensive, 1100 USD - Home, cheap, 600 USD - Home, expensive, 1200 USD - Life, cheap, 100 USD - + Please suggest an insurance given the following budget and requirements: Budget: $1000 Requirements: Car, Home @@ -153,11 +153,11 @@ Running this prompt through an LLM will produce a response like this: ```output , and Life insurance - - Given the budget and requirements, we suggest the following insurance package from ACME Insurance: - - Car, cheap, 500 USD - - Home, cheap, 600 USD - - Life, cheap, 100 USD + + Given the budget and requirements, we suggest the following insurance package from ACME Insurance: + - Car, cheap, 500 USD + - Home, cheap, 600 USD + - Life, cheap, 100 USD Total cost: $1,200 USD ``` @@ -165,13 +165,13 @@ As you can see, it also suggests the Life insurance, which it shouldn't. This re ```text Insurance company: ACME Insurance - Insurance products (cost per month): + Insurance products (cost per month): - type: Car, cheap, cost: 500 USD - - type: Car, expensive, cost: 1100 USD + - type: Car, expensive, cost: 1100 USD - type: Home, cheap, cost: 600 USD - type: Home, expensive, cost: 1200 USD - type: Life, cheap, cost: 100 USD - + Please suggest an insurance given the following budget and requirements: Budget: $1000 restrict choice to types: Car, Home ``` @@ -222,11 +222,11 @@ AI Answer: ```python import flask app = Flask(__name__) - + @app.route('/products') def products(): return 'Products' - + @app.route('/customers') def customers(): return 'Customers' @@ -238,27 +238,27 @@ AI Answer: ```text Sure, here are three suggestions to improve the code: - + Import the correct module: The first line should import Flask instead of flask. The correct import statement should be from flask import Flask. - + Add a main function: Add a if __name__ == '__main__': block to the end of the file to run the application when the file is executed. - + Use f-strings: Use f-strings to format the return statements in the route functions. For example, return 'Products' can be changed to return f'This is the {request.path} page'. ``` ```python from flask import Flask, request - + app = Flask(__name__) - + @app.route('/products') def products(): return f'This is the {request.path} page' - + @app.route('/customers') def customers(): return f'This is the {request.path} page' - + if __name__ == '__main__': app.run() ``` @@ -278,7 +278,7 @@ Repeat 2 and 3 until you've gone through all the parts and you're satisfied with Here's an example prompt: > prompt: How can I create a crisis plan to mitigate a pandemic in 5 steps? -LLM answer: +> LLM answer: ```text 1. Identify the risks @@ -294,7 +294,7 @@ It's identified 5 steps, but can we determine if this is correct? Let's ask the ```text The risks are the following: - + - Loss of life - Loss of business - Loss of reputation @@ -328,34 +328,34 @@ LLMs are nondeterministic by nature, meaning that you will get different results # Import necessary modules import flask from flask import request, jsonify - + # Create a Flask app app = flask.Flask(__name__) - + # Create a route for the API @app.route('/api', methods=['GET']) def api(): # Get the data from the request data = request.args - + # Process the data result = process_data(data) - + # Return the result as JSON return jsonify(result) - + # Function to process the data def process_data(data): # Do something with the data result = data + 1 - + # Return the result return result - + # Run the app if __name__ == '__main__': app.run() - + ``` Running the same prompt again generates a slightly different response: @@ -364,11 +364,11 @@ Running the same prompt again generates a slightly different response: #import necessary packages import flask from flask import request, jsonify - + #create the Flask application app = flask.Flask(__name__) app.config["DEBUG"] = True - + #create a list of books books = [ {'id': 0, @@ -387,18 +387,18 @@ Running the same prompt again generates a slightly different response: 'first_sentence': 'to wound the autumnal city.', 'published': '1975'} ] - + #create an endpoint for the API @app.route('/', methods=['GET']) def home(): return '''

Books API

A prototype API for retrieving books.

''' - + #create an endpoint to return all books @app.route('/api/v1/resources/books/all', methods=['GET']) def api_all(): return jsonify(books) - + #create an endpoint to return a single book @app.route('/api/v1/resources/books', methods=['GET']) def api_id(): @@ -409,20 +409,20 @@ Running the same prompt again generates a slightly different response: id = int(request.args['id']) else: return "Error: No id field provided. Please specify an id." - + #create an empty list for our results results = [] - + #loop through the data and match results that fit the requested ID #IDs are unique, but other fields might return many results for book in books: if book['id'] == id: results.append(book) - + #use the jsonify function from Flask to convert our list of #Python dictionaries to the JSON format return jsonify(results) - + app.run() ``` @@ -442,34 +442,34 @@ Temperature is a value between 0 and 1, where 0 is the most deterministic and 1 #import necessary libraries import flask from flask import request, jsonify - + #create an instance of the Flask class app = flask.Flask(__name__) - + #create an endpoint for the API @app.route('/api/v1/endpoint', methods=['GET']) def api_endpoint(): #get the request data data = request.get_json() - + #process the data result = process_data(data) - + #return the result return jsonify(result) - + #function to process the data def process_data(data): #process the data result = {'result': 'success'} - + #return the result return result - + #run the Flask app if __name__ == '__main__': app.run() - + ``` Running the prompt again gives us this outcome: @@ -478,70 +478,70 @@ Running the prompt again gives us this outcome: #import necessary libraries import flask from flask import request, jsonify - + #create a Flask app app = flask.Flask(__name__) - + #create a route for the API @app.route('/api', methods=['GET']) def api_call(): #get the data from the request data = request.get_json() - + #process the data result = process_data(data) - + #return the result return jsonify(result) - + #define the process_data function def process_data(data): #do something with the data result = data + 1 - + #return the result return result - + #run the app if __name__ == '__main__': app.run() - + ``` -There's only a tiny difference between these two outputs. Let's do the opposite this time, let's set the temperature to 0.9: +There's only a tiny difference between these two outputs. Let's do the opposite this time, let's set the temperature to 0.9: ```python # Import necessary libraries import flask from flask import request, jsonify - + # Create a Flask app app = flask.Flask(__name__) - + # Create a route for the API @app.route('/api', methods=['GET']) def api_call(): # Get the data from the request data = request.args - + # Process the data result = process_data(data) - + # Return the result return jsonify(result) - + # Function to process the data def process_data(data): # Do something with the data result = data + 1 - + # Return the result return result - + # Run the app if __name__ == '__main__': app.run() - + ``` and the second attempt at 0.9 as temperature value: @@ -549,34 +549,34 @@ and the second attempt at 0.9 as temperature value: ```python import flask from flask import request, jsonify - + # create the Flask app app = flask.Flask(__name__) app.config['DEBUG'] = True - + # create some test data books = [ {'id': 0, 'title': 'A Fire Upon The Deep', 'author': 'Vernor Vinge', 'first_sentence': 'The coldsleep itself was dreamless.', 'year_published': '1992'}, {'id': 1, 'title': 'The Ones Who Walk Away From Omelas', 'author': 'Ursula K. Le Guin', 'first_sentence': 'With a clamor of bells that set the swallows soaring, the Festival of Summer came to the city Omelas, bright-towered by the sea.', 'published': '1973'}, {'id': 2, 'title': 'Dhalgren', 'author': 'Samuel R. Delany', 'first_sentence': 'to wound the autumnal city.', 'published': '1975'} ] - + # create an endpoint @app.route('/', methods=['GET']) def home(): return '''

Welcome to our book API!

''' - + @app.route('/api/v1/resources/books - + ``` As you can see, the results couldn't be more varied. -> Note, that there are more parameters you can change to vary the output, like top-k, top-p, repetition penalty, length penalty and diversity penalty but these are outside the scope of this curriculum. +> Note, that there are more parameters you can change to vary the output, like top-k, top-p, repetition penalty, length penalty and diversity penalty but these are outside the scope of this curriculum. ## Good practices -There are many practices you can apply to try to get what you want. You will find your own style as you use prompting more and more. +There are many practices you can apply to try to get what you want. You will find your own style as you use prompting more and more. Additionally to the techniques we've covered, there are some good practices to consider when prompting an LLM. @@ -586,7 +586,7 @@ Here are some good practices to consider: - Limit the output. If you want a specific number of items or a specific length, specify it. - **Specify both what and how**. Remember to mention both what you want and how you want it, for example "Create a Python Web API with routes products and customers, divide it into 3 files". - **Use templates**. Often, you will want to enrich your prompts with data from your company. Use templates to do this. Templates can have variables that you replace with actual data. -- **Spell correctly**. LLMs might provide you with a correct response, but if you spell correctly, you will get a better response. +- **Spell correctly**. LLMs might provide you with a correct response, but if you spell correctly, you will get a better response. ## Assignment @@ -594,14 +594,14 @@ Here's code in Python showing how to build a simple API using Flask: ```python from flask import Flask, request - + app = Flask(__name__) - + @app.route('/') def hello(): name = request.args.get('name', 'World') return f'Hello, {name}!' - + if __name__ == '__main__': app.run() ``` @@ -629,7 +629,7 @@ A: 1, because chain-of-thought is about showing the LLM how to solve a problem b ## 🚀 Challenge -You just used the self-refine technique in the assignment. Take any program you built and consider what improvements you would want to apply to it. Now use the self-refine technique to apply the proposed changes. What did you think the result, better or worse? +You just used the self-refine technique in the assignment. Take any program you built and consider what improvements you would want to apply to it. Now use the self-refine technique to apply the proposed changes. What did you think the result, better or worse? ## Great Work! Continue Your Learning diff --git a/05-advanced-prompts/translations/cn/README.md b/05-advanced-prompts/translations/cn/README.md index 624831984..ef3ef862a 100644 --- a/05-advanced-prompts/translations/cn/README.md +++ b/05-advanced-prompts/translations/cn/README.md @@ -1,6 +1,6 @@ # 第五章:创建高级的提示工程技巧 -[![Creating Advanced Prompts](../../images/05-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/32GBH6BTWZQ?WT.mc_id=academic-105485-koreyst) +[![Creating Advanced Prompts](../../images/05-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=f3615e46-dbca-477c-832d-21eef95cd9f3?WT.mc_id=academic-105485-koreyst) 让我们回顾一下上一章的一些知识: @@ -81,11 +81,11 @@ LLM 给出的答案为 5,这是不正确的。 根据计算结果 (5 -3 -2 + 1 = 1),正确答案是 1 个苹果。 -那么我们怎样才能教LLM正确地做到这一点呢? +那么我们怎样才能教 LLM 正确地做到这一点呢? 让我们尝试一下思维链。 应用思维链意味着: -1. 给LLM一个类似的例子。 +1. 给 LLM 一个类似的例子。 2. 展示计算结果,以及如何正确计算。 3. 提供原始提示。 @@ -108,7 +108,7 @@ LLM 给出的答案为 5,这是不正确的。 根据计算结果 (5 -3 -2 + 1 ```text {{company}}: {{company_name}} - {{products}}: + {{products}}: {{products_list}} Please suggest an insurance given the following budget and requirements: Budget: {{budget}} @@ -121,13 +121,13 @@ LLM 给出的答案为 5,这是不正确的。 根据计算结果 (5 -3 -2 + 1 ```text Insurance company: ACME Insurance - Insurance products (cost per month): + Insurance products (cost per month): - Car, cheap, 500 USD - - Car, expensive, 1100 USD + - Car, expensive, 1100 USD - Home, cheap, 600 USD - Home, expensive, 1200 USD - Life, cheap, 100 USD - + Please suggest an insurance given the following budget and requirements: Budget: $1000 Requirements: Car, Home @@ -137,11 +137,11 @@ LLM 给出的答案为 5,这是不正确的。 根据计算结果 (5 -3 -2 + 1 ```output , and Life insurance - - Given the budget and requirements, we suggest the following insurance package from ACME Insurance: - - Car, cheap, 500 USD - - Home, cheap, 600 USD - - Life, cheap, 100 USD + + Given the budget and requirements, we suggest the following insurance package from ACME Insurance: + - Car, cheap, 500 USD + - Home, cheap, 600 USD + - Life, cheap, 100 USD Total cost: $1,200 USD ``` @@ -149,13 +149,13 @@ LLM 给出的答案为 5,这是不正确的。 根据计算结果 (5 -3 -2 + 1 ```text Insurance company: ACME Insurance - Insurance products (cost per month): + Insurance products (cost per month): - type: Car, cheap, cost: 500 USD - - type: Car, expensive, cost: 1100 USD + - type: Car, expensive, cost: 1100 USD - type: Home, cheap, cost: 600 USD - type: Home, expensive, cost: 1200 USD - type: Life, cheap, cost: 100 USD - + Please suggest an insurance given the following budget and requirements: Budget: $1000 restrict choice to types: Car, Home ``` @@ -168,7 +168,7 @@ LLM 给出的答案为 5,这是不正确的。 根据计算结果 (5 -3 -2 + 1 Given the budget and requirements, we suggest the Car, Cheap insurance product which costs 500 USD per month. ``` -这个例子的目的是表明,即使我们使用像_生成知识_这样的基本技术,我们仍然需要在大多数情况下优化提示以获得所需的结果。 +这个例子的目的是表明,即使我们使用像*生成知识*这样的基本技术,我们仍然需要在大多数情况下优化提示以获得所需的结果。 ### 从少到多 @@ -186,12 +186,12 @@ LLM 给出的答案为 5,这是不正确的。 根据计算结果 (5 -3 -2 + 1 ### 自我完善,质疑结果 -对于生成式人工智能和 LLMs,你不能相信其输出。 你需要验证一下。 毕竟, LLMs只是向您展示下一个最有可能说的话,而不是正确的内容。 因此,一个好主意是要求 LLMs 自我批评,这引导我们自我完善技术。 +对于生成式人工智能和 LLMs,你不能相信其输出。 你需要验证一下。 毕竟, LLMs 只是向您展示下一个最有可能说的话,而不是正确的内容。 因此,一个好主意是要求 LLMs 自我批评,这引导我们自我完善技术。 其工作原理是按照以下步骤操作: -1. 要求LLM解决问题的初始提示 -2. LLM产生答案 +1. 要求 LLM 解决问题的初始提示 +2. LLM 产生答案 3. 质疑答案并要求人工智能改进 4. LLM 再次回答,这次考虑了质疑并提出了解决方案 @@ -206,11 +206,11 @@ AI Answer: ```python import flask app = Flask(__name__) - + @app.route('/products') def products(): return 'Products' - + @app.route('/customers') def customers(): return 'Customers' @@ -222,27 +222,27 @@ AI Answer: ```text Sure, here are three suggestions to improve the code: - + Import the correct module: The first line should import Flask instead of flask. The correct import statement should be from flask import Flask. - + Add a main function: Add a if __name__ == '__main__': block to the end of the file to run the application when the file is executed. - + Use f-strings: Use f-strings to format the return statements in the route functions. For example, return 'Products' can be changed to return f'This is the {request.path} page'. ``` ```python from flask import Flask, request - + app = Flask(__name__) - + @app.route('/products') def products(): return f'This is the {request.path} page' - + @app.route('/customers') def customers(): return f'This is the {request.path} page' - + if __name__ == '__main__': app.run() ``` @@ -254,13 +254,13 @@ AI Answer: 多维度的提示是一种类似于自我完善的技术,但它更多的是要求 LLMs 解释自己。 目标是减少 LLMs 输出不一致,以确保得出正确的答案。 要遵循的工作流程是: 1. 请 LLM 回答问题 -2. 对于答案的每一部分,请LLM更深入地解释。 +2. 对于答案的每一部分,请 LLM 更深入地解释。 3. 如果存在不一致,则丢弃不一致的部分。 重复 2 和 3,直到您完成所有部分并对答案感到满意为止。 > prompt: How can I create a crisis plan to mitigate for a pandemic in 5 steps? -LLM answer: +> LLM answer: ```text 1. Identify the risks @@ -270,13 +270,13 @@ LLM answer: 1. Identify the communication channels ``` -它确定了 5 个步骤,但是我们可以确定这是否正确吗? 让我们请LLM解释一下每一步: +它确定了 5 个步骤,但是我们可以确定这是否正确吗? 让我们请 LLM 解释一下每一步: > prompt: Explain the first step in more detail, what are the risks in detail with a pandemic? ```text The risks are the following: - + - Loss of life - Loss of business - Loss of reputation @@ -310,34 +310,34 @@ LLM 本质上是不确定的,这意味着每次运行相同的提示时都会 # Import necessary modules import flask from flask import request, jsonify - + # Create a Flask app app = flask.Flask(__name__) - + # Create a route for the API @app.route('/api', methods=['GET']) def api(): # Get the data from the request data = request.args - + # Process the data result = process_data(data) - + # Return the result as JSON return jsonify(result) - + # Function to process the data def process_data(data): # Do something with the data result = data + 1 - + # Return the result return result - + # Run the app if __name__ == '__main__': app.run() - + ``` 再次运行相同的提示会生成略有不同的响应: @@ -346,11 +346,11 @@ LLM 本质上是不确定的,这意味着每次运行相同的提示时都会 #import necessary packages import flask from flask import request, jsonify - + #create the Flask application app = flask.Flask(__name__) app.config["DEBUG"] = True - + #create a list of books books = [ {'id': 0, @@ -369,18 +369,18 @@ LLM 本质上是不确定的,这意味着每次运行相同的提示时都会 'first_sentence': 'to wound the autumnal city.', 'published': '1975'} ] - + #create an endpoint for the API @app.route('/', methods=['GET']) def home(): return '''

Books API

A prototype API for retrieving books.

''' - + #create an endpoint to return all books @app.route('/api/v1/resources/books/all', methods=['GET']) def api_all(): return jsonify(books) - + #create an endpoint to return a single book @app.route('/api/v1/resources/books', methods=['GET']) def api_id(): @@ -391,20 +391,20 @@ LLM 本质上是不确定的,这意味着每次运行相同的提示时都会 id = int(request.args['id']) else: return "Error: No id field provided. Please specify an id." - + #create an empty list for our results results = [] - + #loop through the data and match results that fit the requested ID #IDs are unique, but other fields might return many results for book in books: if book['id'] == id: results.append(book) - + #use the jsonify function from Flask to convert our list of #Python dictionaries to the JSON format return jsonify(results) - + app.run() ``` @@ -424,34 +424,34 @@ LLM 本质上是不确定的,这意味着每次运行相同的提示时都会 #import necessary libraries import flask from flask import request, jsonify - + #create an instance of the Flask class app = flask.Flask(__name__) - + #create an endpoint for the API @app.route('/api/v1/endpoint', methods=['GET']) def api_endpoint(): #get the request data data = request.get_json() - + #process the data result = process_data(data) - + #return the result return jsonify(result) - + #function to process the data def process_data(data): #process the data result = {'result': 'success'} - + #return the result return result - + #run the Flask app if __name__ == '__main__': app.run() - + ``` 再次运行提示符会给我们这样的结果: @@ -460,34 +460,34 @@ LLM 本质上是不确定的,这意味着每次运行相同的提示时都会 #import necessary libraries import flask from flask import request, jsonify - + #create a Flask app app = flask.Flask(__name__) - + #create a route for the API @app.route('/api', methods=['GET']) def api_call(): #get the data from the request data = request.get_json() - + #process the data result = process_data(data) - + #return the result return jsonify(result) - + #define the process_data function def process_data(data): #do something with the data result = data + 1 - + #return the result return result - + #run the app if __name__ == '__main__': app.run() - + ``` 这两个输出之间只有很小的差异。 这次我们反其道而行之,将温度设置为 0.9: @@ -496,34 +496,34 @@ LLM 本质上是不确定的,这意味着每次运行相同的提示时都会 # Import necessary libraries import flask from flask import request, jsonify - + # Create a Flask app app = flask.Flask(__name__) - + # Create a route for the API @app.route('/api', methods=['GET']) def api_call(): # Get the data from the request data = request.args - + # Process the data result = process_data(data) - + # Return the result return jsonify(result) - + # Function to process the data def process_data(data): # Do something with the data result = data + 1 - + # Return the result return result - + # Run the app if __name__ == '__main__': app.run() - + ``` 第二次尝试使用 0.9 作为温度值: @@ -531,30 +531,30 @@ LLM 本质上是不确定的,这意味着每次运行相同的提示时都会 ```python import flask from flask import request, jsonify - + # create the Flask app app = flask.Flask(__name__) app.config['DEBUG'] = True - + # create some test data books = [ {'id': 0, 'title': 'A Fire Upon The Deep', 'author': 'Vernor Vinge', 'first_sentence': 'The coldsleep itself was dreamless.', 'year_published': '1992'}, {'id': 1, 'title': 'The Ones Who Walk Away From Omelas', 'author': 'Ursula K. Le Guin', 'first_sentence': 'With a clamor of bells that set the swallows soaring, the Festival of Summer came to the city Omelas, bright-towered by the sea.', 'published': '1973'}, {'id': 2, 'title': 'Dhalgren', 'author': 'Samuel R. Delany', 'first_sentence': 'to wound the autumnal city.', 'published': '1975'} ] - + # create an endpoint @app.route('/', methods=['GET']) def home(): return '''

Welcome to our book API!

''' - + @app.route('/api/v1/resources/books - + ``` 正如您所看到的,结果各式各样。 -> 请注意,您可以更改更多参数来改变输出,例如 top-k, top-p, repetition penalty, length penalty and diversity penalty ,但这些不属于本课程的范围。 +> 请注意,您可以更改更多参数来改变输出,例如 top-k, top-p, repetition penalty, length penalty and diversity penalty ,但这些不属于本课程的范围。 ## 最佳实践 @@ -576,14 +576,14 @@ LLM 本质上是不确定的,这意味着每次运行相同的提示时都会 ```python from flask import Flask, request - + app = Flask(__name__) - + @app.route('/') def hello(): name = request.args.get('name', 'World') return f'Hello, {name}!' - + if __name__ == '__main__': app.run() ``` @@ -617,4 +617,4 @@ LLM 本质上是不确定的,这意味着每次运行相同的提示时都会 想要了解有关创建高级提示的更多信息? 转至[进阶学习的页面](../../../13-continued-learning/translations/cn/README.md?WT.mc_id=academic-105485-koreyst) 查找有关此主章节的其他学习资源。 -前往第六章,我们将应用提示工程知识[构建文本生成应用程序](../../../06-text-generation-apps/translations/cn/README.md?WT.mc_id=academic-105485-koreyst) +前往第六章,我们将应用提示工程知识[构建文本生成应用程序](../../../06-text-generation-apps/translations/cn/README.md?WT.mc_id=academic-105485-koreyst) diff --git a/05-advanced-prompts/translations/ja-jp/README.md b/05-advanced-prompts/translations/ja-jp/README.md index 9f1c3659f..3da392471 100644 --- a/05-advanced-prompts/translations/ja-jp/README.md +++ b/05-advanced-prompts/translations/ja-jp/README.md @@ -1,6 +1,6 @@ # 高度なプロンプトの作成 -[![高度なプロンプトの作成](../../images/05-lesson-banner.png?WT.mc_id=academic-105485-yoterada)](https://youtu.be/32GBH6BTWZQ?WT.mc_id=academic-105485-yoterada) +[![高度なプロンプトの作成](../../images/05-lesson-banner.png?WT.mc_id=academic-105485-yoterada)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=f3615e46-dbca-477c-832d-21eef95cd9f3?WT.mc_id=academic-105485-yoterada) 前章までの学習内容をいくつかおさらいしましょう。 @@ -71,7 +71,7 @@ LLM のプロンプトで利用可能な、基本的なテクニックがいく このプロンプトはとても簡単で、一つのプロンプトといくつかの例から構成します。 この手法は、LLM について学び始めたときに最初に使用すると良い手法です。次に例を示します。 -- プロンプト: 「代数とは何ですか?」 +- プロンプト: 「代数とは何ですか?」 - 回答: 「数学の一分野で、数の代わりに文字を用いて方程式の解法などを研究する学問です。」 > [!NOTE] @@ -82,22 +82,22 @@ LLM のプロンプトで利用可能な、基本的なテクニックがいく Chain-of-thought (思考の連鎖) は、非常に興味深い手法で、一つの問題をいくつか複数の段階に分けて、LLM の回答を誘導する手法です。LLM が、何をどうやって処理を行うべきか理解できるように指示する手法です。次の例を考えてみてください。Chain-of-thought を用いた場合と用いない場合の両方の例をご覧ください。まずは、Chain-of-thought を用いない例を見てみましょう。 - - プロンプト: 「アリスはリンゴを 5 個持っていて、3 個を捨て、2 個をボブに渡し、ボブは 1 個を返します。アリスはリンゴを何個持っていますか?」 - - 回答: 5 + - プロンプト: 「アリスはリンゴを 5 個持っていて、3 個を捨て、2 個をボブに渡し、ボブは 1 個を返します。アリスはリンゴを何個持っていますか?」 + - 回答: 5 -LLM は 5 と答えますが、これは不正解です。正しい答えはリンゴ 1 個で、計算式は (5 -3 -2 + 1 = 1) になります。 +LLM は 5 と答えますが、これは不正解です。正しい答えはリンゴ 1 個で、計算式は (5 -3 -2 + 1 = 1) になります。 -では、どうやって LLM がこれを正しく計算できるように伝えられるのでしょうか? +では、どうやって LLM がこれを正しく計算できるように伝えられるのでしょうか? -次に、Chain-of-thought を試してみましょう。Chain-of-thought を適用するためには: +次に、Chain-of-thought を試してみましょう。Chain-of-thought を適用するためには: -1. LLM に類似の例を示す。 -2. 計算式と、それを正しく計算する方法を示す。 -3. 元のプロンプトを提供する。 +1. LLM に類似の例を示す。 +2. 計算式と、それを正しく計算する方法を示す。 +3. 元のプロンプトを提供する。 -という手順を踏みます。以下にその方法を示します。 +という手順を踏みます。以下にその方法を示します。 - - プロンプト: 「リサはリンゴを7個持っていて、1個を捨て、4個をバートに渡し、バートは1個を返します。計算は次の通り: + - プロンプト: 「リサはリンゴを7個持っていて、1個を捨て、4個をバートに渡し、バートは1個を返します。計算は次の通り: 7 - 1 = 6 6 - 4 = 2 2 + 1 = 3 @@ -119,81 +119,81 @@ LLM は 5 と答えますが、これは不正解です。正しい答えはリ 例えば、保険業界であれば、プロンプトは以下のようになります。 ```text -{{company}}: {{company_name}} -{{products}}: -{{products_list}} -以下の予算と要件を考慮し、保険を提案してください: -予算: {{budget}} -要件: {{requirements}} +{{company}}: {{company_name}} +{{products}}: +{{products_list}} +以下の予算と要件を考慮し、保険を提案してください: +予算: {{budget}} +要件: {{requirements}} ``` -上記は、テンプレートを使用してプロンプトをどのように作成するかを示しています。テンプレートには、`{{variable}}`という形式で表現する変数がいくつかあり、これらは会社の API から取得した実際の値に置き換えます。 +上記は、テンプレートを使用してプロンプトをどのように作成するかを示しています。テンプレートには、`{{variable}}`という形式で表現する変数がいくつかあり、これらは会社の API から取得した実際の値に置き換えます。 -以下は、変数を自社の情報に置き換えた後のプロンプト例です。 +以下は、変数を自社の情報に置き換えた後のプロンプト例です。 ```text -保険会社: ACME保険 -保険商品(月額料金): -- 自動車保険, 安価, 500ドル -- 自動車保険, 高価, 1100ドル -- 家財保険, 安価, 600ドル -- 家財保険, 高価, 1200ドル -- 生命保険, 安価, 100ドル - -以下の予算と要件を考慮した上で、保険を提案してください: -予算: $1000 -要件: 自動車保険, 家財保険 +保険会社: ACME保険 +保険商品(月額料金): +- 自動車保険, 安価, 500ドル +- 自動車保険, 高価, 1100ドル +- 家財保険, 安価, 600ドル +- 家財保険, 高価, 1200ドル +- 生命保険, 安価, 100ドル + +以下の予算と要件を考慮した上で、保険を提案してください: +予算: $1000 +要件: 自動車保険, 家財保険 ``` このプロンプトを LLM(Large Language Model)を実行すると、以下のような回答が得られます。 ```output -生命保険も含めて - -予算と要件を考慮した上で、ACME保険から以下の保険パッケージを提案します: -- 自動車保険, 安価, 500ドル -- 家財保険, 安価, 600ドル -- 生命保険, 安価, 100ドル -合計費用: $1,200ドル +生命保険も含めて + +予算と要件を考慮した上で、ACME保険から以下の保険パッケージを提案します: +- 自動車保険, 安価, 500ドル +- 家財保険, 安価, 600ドル +- 生命保険, 安価, 100ドル +合計費用: $1,200ドル ``` 結果は、生命保険も提案されましたが、これは本来求めていない結果です。そして、プロンプトの最適化が必要だというのを示しています。プロンプトで何を許可するのかを、より明確にする必要があります。試行錯誤を繰り返した結果、以下のようなプロンプトにたどり着きました。 ```text -保険会社: ACME保険 -保険商品(月額料金): -- 種類: 自動車保険, 安価, 費用: 500ドル -- 種類: 自動車保険, 高価, 費用: 1100ドル -- 種類: 家財保険, 安価, 費用: 600ドル -- 種類: 家財保険, 高価, 費用: 1200ドル -- 種類: 生命保険, 安価, 費用: 100ドル - -以下の予算と要件を考慮した上で、保険を提案してください: -予算: $1000 選択を以下の種類に限定: 自動車保険, 家財保険 -``` +保険会社: ACME保険 +保険商品(月額料金): +- 種類: 自動車保険, 安価, 費用: 500ドル +- 種類: 自動車保険, 高価, 費用: 1100ドル +- 種類: 家財保険, 安価, 費用: 600ドル +- 種類: 家財保険, 高価, 費用: 1200ドル +- 種類: 生命保険, 安価, 費用: 100ドル + +以下の予算と要件を考慮した上で、保険を提案してください: +予算: $1000 選択を以下の種類に限定: 自動車保険, 家財保険 +``` 「種類」と「費用」を追加し、「限定」というキーワードを使用し、LLM が何を求めているかを理解するのに役立ちます。 これにより、以下のような回答が得られました。 ```output -予算と要件を考慮した上で、月額500ドルの自動車保険(安価)を提案します。 +予算と要件を考慮した上で、月額500ドルの自動車保険(安価)を提案します。 ``` この例から、Generated knowledge という手法を使用しても、期待する結果を得るためには、多くの場合でプロンプトの最適化が必要だとわかります。 ### Least-to-most -「Least-to-most (最小から最大) プロンプト」は、大きな問題を小さな問題に分割する手法です。これにより、LLM(Language Model)が大きな問題をどのように「解決」するかをガイドするのに役立ちます。良い例としては、データサイエンスがあります。LLM に以下のように問題を分割するように依頼できます。 +「Least-to-most (最小から最大) プロンプト」は、大きな問題を小さな問題に分割する手法です。これにより、LLM(Language Model)が大きな問題をどのように「解決」するかをガイドするのに役立ちます。良い例としては、データサイエンスがあります。LLM に以下のように問題を分割するように依頼できます。 -> プロンプト: 5つのステップで、データサイエンスを実行する方法は何ですか? +> プロンプト: 5 つのステップで、データサイエンスを実行する方法は何ですか? -AI アシスタントは以下のように回答します。 +AI アシスタントは以下のように回答します。 -1. データを収集する -2. データをクリーニングする -3. データを分析する -4. データをプロットする +1. データを収集する +2. データをクリーニングする +3. データを分析する +4. データをプロットする 5. データを提示する > [!TIP] @@ -202,35 +202,35 @@ AI アシスタントは以下のように回答します。 > Chain of thought の場合は、問題の解き方をプロンプト内で具体的に指示するために、仮に解き方で記載した手順よりも難しい問題が与えられた場合は、時間を要するかもしくは回答できません。 > 一方で、Least-to-most は、複雑な問題をより簡単なサブ問題に分解し、それらを順番に解いていきます。サブ問題を解く際には、前に解いたサブ問題の答えを利用します。 > たとえば、難しい問題を解くために、最初にこの問題の解き方を教えてくださいという、最小の簡単な質問をします。その結果、AI はまず、その問題の解き方を考えて教えてくれます。この問題の解き方の情報を会話履歴に含めたままで、次に、オリジナルの問題をもう一度問い合わせします。すると、会話履歴に含まれる、解き方の情報を使って、オリジナルの問題を解き回答できます。そのため、Chain of thought よりも、より難しい問題を解けるようになります。 -> 解き方 +>  解き方 > -> 1. 複雑な問題を一連の単純なサブ問題に分解する +> 1. 複雑な問題を一連の単純なサブ問題に分解する > 2. サブ質問のそれぞれを、前の回答を元に解く -> 参考情報: [1. Least to Most Prompting](https://cobusgreyling.medium.com/least-to-most-prompting-b37ed2e19859?WT.mc_id=academic-105485-yoterada), [2. LEAST-TO-MOST PROMPTING ENABLES COMPLEX REASONING IN LARGE LANGUAGE MODELS (Original Paper)](https://arxiv.org/pdf/2205.10625.pdf?WT.mc_id=academic-105485-yoterada) +> 参考情報: [1. Least to Most Prompting](https://cobusgreyling.medium.com/least-to-most-prompting-b37ed2e19859?WT.mc_id=academic-105485-yoterada), [2. LEAST-TO-MOST PROMPTING ENABLES COMPLEX REASONING IN LARGE LANGUAGE MODELS (Original Paper)](https://arxiv.org/pdf/2205.10625.pdf?WT.mc_id=academic-105485-yoterada) ### Self-refine, critique the results 生成 AI や LLM を使用する際、その出力をそのまま信じてはいけません。必ず確認が必要です。なぜなら、LLM は何が正しいかではなく、次に出てくる可能性が最も高いものを提示しているだけだからです。そのため、LLM に自己評価を求め、それによって自己改善するという手法が有効です。 -この手法は以下の手順で行います: +この手法は以下の手順で行います: -1. 最初に LLM に問題解決を求める -2. LLM が回答する -3. その回答を評価し、AI に改善を求める -4. LLM が再度回答し、今度は評価を考慮して自身が考えた解決策を提案する +1. 最初に LLM に問題解決を求める +2. LLM が回答する +3. その回答を評価し、AI に改善を求める +4. LLM が再度回答し、今度は評価を考慮して自身が考えた解決策を提案する このプロセスは何度でも繰り返しができます。 以下にこの手法を使用した例を示します: -> プロンプト: "製品と顧客を持つ Python Web API を作成してください" +> プロンプト: "製品と顧客を持つ  Python Web API  を作成してください" -AIの回答: +AI の回答: ```python import flask app = Flask(__name__) - + @app.route('/products') def products(): return 'Products' @@ -240,9 +240,9 @@ def customers(): return 'Customers' ``` -> プロンプト: "上記のコードの改善点を3つ提案してください" +> プロンプト: "上記のコードの改善点を 3 つ提案してください" -AIの回答: +AI の回答: ```text もちろんです、コードを改善するための3つの提案をします: @@ -254,12 +254,11 @@ AIの回答: f-strings を使用する:ルート関数の return 文をフォーマットするために f-strings を使用します。例えば、'Products' を返す代わりに、f'This is the {request.path} page' を返すように変更できます。 ``` - ```python from flask import Flask, request - + app = Flask(__name__) - + @app.route('/products') def products(): return f'This is the {request.path} page' @@ -276,19 +275,19 @@ if __name__ == '__main__': ### Maieutic prompting -Maieutic プロンプトは、self-refine (自己改善) と似た手法ですが、こちらは LLM に自身で説明を求めるものです。LLM の出力における不整合を減らし、正確な回答を取得するのを目指します。そのための手順は以下のようになります。 +Maieutic プロンプトは、self-refine (自己改善) と似た手法ですが、こちらは LLM に自身で説明を求めるものです。LLM の出力における不整合を減らし、正確な回答を取得するのを目指します。そのための手順は以下のようになります。 1. LLM に質問に答えるよう求める 2. 答えの各部分について、LLM に詳しく説明させる -3. 不整合がある場合、その部分は排除する +3. 不整合がある場合、その部分は排除する これら全ての内容を検討し、回答に満足するまで 2 と 3 の手順を繰り返します。 以下に、例を示します。 -> プロンプト: パンデミックを緩和するための危機対策を5つの段階でどのように作成すればよいか教えてください。 +> プロンプト: パンデミックを緩和するための危機対策を 5 つの段階でどのように作成すればよいか教えてください。 -LLMの回答: +LLM の回答: ```text 1. リスクを特定する @@ -302,31 +301,31 @@ LLMの回答: > プロンプト: 最初のステップを詳しく説明してください。具体的には、パンデミックにおけるリスクは何ですか? -```text -リスクは以下の通りです: +```text +リスクは以下の通りです: - 生命の喪失 - ビジネスの喪失 - 評判の喪失 - 顧客の喪失 - 従業員の喪失 -``` +``` ここで、LLM が正しいかどうかを確認するため、最初のリスクについて詳しく説明させ、それが上記の答えと一致するか確認します。 > プロンプト: パンデミックにおいて、最も大きなリスクは何でしょうか?また、その理由は何ですか? -```text -最も大きなリスクは生命の喪失で、それは最も重要な守るべきものです。 +```text +最も大きなリスクは生命の喪失で、それは最も重要な守るべきものです。 ``` -> パンデミックにおけるリスクの中で、最も大きな2つのリスクは何だと思いますか? +> パンデミックにおけるリスクの中で、最も大きな 2 つのリスクは何だと思いますか? ```text 最も大きな2つのリスクは、生命の喪失とビジネスの喪失です。 ``` -ここで、LLM は一貫性を保ち、「生命」と「ビジネス」を最大の2つのリスクとして挙げています。これにより、次のステップに進み、かなりの自信を持てます。ただし、LLM を盲目的に信頼するのではなく、常にその出力を検証するのが重要です。 +ここで、LLM  は一貫性を保ち、「生命」と「ビジネス」を最大の 2 つのリスクとして挙げています。これにより、次のステップに進み、かなりの自信を持てます。ただし、LLM  を盲目的に信頼するのではなく、常にその出力を検証するのが重要です。 ## Vary your output @@ -334,8 +333,8 @@ LLM は本質的に非確実な性質を持っているため、同じプロン > "Python Web API のコードを生成してください" -```python -# 必要なモジュールをインポートする +```python +# 必要なモジュールをインポートする import flask from flask import request, jsonify @@ -479,7 +478,7 @@ if __name__ == '__main__': app.run() ``` -プロンプトを再度実行すると、次の結果が得られます。 +プロンプトを再度実行すると、次の結果が得られます。 ```python #import necessary libraries @@ -514,7 +513,7 @@ if __name__ == '__main__': app.run() ``` -これら 2 つの出力結果はわずかな違いしかありません。今度は逆に、温度を0.9に設定してみましょう。 +これら 2 つの出力結果はわずかな違いしかありません。今度は逆に、温度を 0.9 に設定してみましょう。 ```python # Import necessary libraries @@ -586,7 +585,7 @@ def home(): 以下に、そのおすすめの方法をいくつか紹介します。 -- **コンテキストを明確にする**:コンテキスト(文脈)は重要で、ドメインやトピックなど、具体的に指定するほど良い結果が得られます。 +- **コンテキストを明確にする**:コンテキスト(文脈)は重要で、ドメインやトピックなど、具体的に指定するほど良い結果が得られます。 - **出力を制限する**:特定の項目数や特定の長さが必要な場合は、それを明示的に指定しましょう。 - **「何を」、「どうするか」を明確にする**:例えば、「Python の Web API を作成し、商品と顧客を設定し、それを 3 つのファイルに分けてください」といった、具体的な指示を出すようにしてください - **テンプレートを活用する**:自社データをプロンプトに反映させたい場合も多いと思います。その際に、テンプレートを活用しましょう。テンプレートには、実際のデータに置き換えられる変数を設定できます。 diff --git a/05-advanced-prompts/translations/pt-br/README.md b/05-advanced-prompts/translations/pt-br/README.md index 711d574ab..e8b5b5a58 100644 --- a/05-advanced-prompts/translations/pt-br/README.md +++ b/05-advanced-prompts/translations/pt-br/README.md @@ -1,6 +1,6 @@ # Criando prompts avançados -[![Creating Advanced Prompts](../../images/05-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/32GBH6BTWZQ?WT.mc_id=academic-105485-koreyst) +[![Creating Advanced Prompts](../../images/05-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=f3615e46-dbca-477c-832d-21eef95cd9f3?WT.mc_id=academic-105485-koreyst) Vamos recapitular algumas aprendizagens do capítulo anterior: @@ -110,7 +110,7 @@ Como exemplo, é assim que sua solicitação pode parecer se você estiver no ra ```text {{company}}: {{company_name}} - {{products}}: + {{products}}: {{products_list}} Please suggest an insurance given the following budget and requirements: Budget: {{budget}} @@ -123,13 +123,13 @@ Aqui está um exemplo de como a solicitação pode parecer depois que as variáv ```text Insurance company: ACME Insurance - Insurance products (cost per month): + Insurance products (cost per month): - Car, cheap, 500 USD - - Car, expensive, 1100 USD + - Car, expensive, 1100 USD - Home, cheap, 600 USD - Home, expensive, 1200 USD - Life, cheap, 100 USD - + Please suggest an insurance given the following budget and requirements: Budget: $1000 Requirements: Car, Home @@ -139,11 +139,11 @@ Executando esta solicitação em um Modelo de Linguagem com Aprendizado Profundo ```output , and Life insurance - - Given the budget and requirements, we suggest the following insurance package from ACME Insurance: - - Car, cheap, 500 USD - - Home, cheap, 600 USD - - Life, cheap, 100 USD + + Given the budget and requirements, we suggest the following insurance package from ACME Insurance: + - Car, cheap, 500 USD + - Home, cheap, 600 USD + - Life, cheap, 100 USD Total cost: $1,200 USD ``` @@ -151,13 +151,13 @@ Como você pode ver, também sugere o seguro de vida, o que não deveria acontec ```text Insurance company: ACME Insurance - Insurance products (cost per month): + Insurance products (cost per month): - type: Car, cheap, cost: 500 USD - - type: Car, expensive, cost: 1100 USD + - type: Car, expensive, cost: 1100 USD - type: Home, cheap, cost: 600 USD - type: Home, expensive, cost: 1200 USD - type: Life, cheap, cost: 100 USD - + Please suggest an insurance given the following budget and requirements: Budget: $1000 restrict choice to types: Car, Home ``` @@ -237,11 +237,11 @@ Resposta da IA: ```python import flask app = Flask(__name__) - + @app.route('/products') def products(): return 'Products' - + @app.route('/customers') def customers(): return 'Customers' @@ -253,27 +253,27 @@ Resposta da IA: ```text Sure, here are three suggestions to improve the code: - + Import the correct module: The first line should import Flask instead of flask. The correct import statement should be from flask import Flask. - + Add a main function: Add a if __name__ == '__main__': block to the end of the file to run the application when the file is executed. - + Use f-strings: Use f-strings to format the return statements in the route functions. For example, return 'Products' can be changed to return f'This is the {request.path} page'. ``` ```python from flask import Flask, request - + app = Flask(__name__) - + @app.route('/products') def products(): return f'This is the {request.path} page' - + @app.route('/customers') def customers(): return f'This is the {request.path} page' - + if __name__ == '__main__': app.run() ``` @@ -305,7 +305,7 @@ Repita os passos 2 e 3 até passar por todas as partes e ficar satisfeito com a Aqui está um exemplo de um prompt: > prompt: How can I create a crisis plan to mitigate a pandemic in 5 steps? -LLM answer: +> LLM answer: ```text 1. Identify the risks @@ -321,7 +321,7 @@ São identificadas 5 etapas, mas podemos determinar se isso está correto? Vamos ```text The risks are the following: - + - Loss of life - Loss of business - Loss of reputation @@ -355,30 +355,30 @@ Os LLMs são, por natureza, não-determinísticos, o que significa que você obt # Import necessary modules import flask from flask import request, jsonify - + # Create a Flask app app = flask.Flask(__name__) - + # Create a route for the API @app.route('/api', methods=['GET']) def api(): # Get the data from the request data = request.args - + # Process the data result = process_data(data) - + # Return the result as JSON return jsonify(result) - + # Function to process the data def process_data(data): # Do something with the data result = data + 1 - + # Return the result return result - + # Run the app if __name__ == '__main__': app.run() @@ -390,11 +390,11 @@ Execute o mesmo prompt novamente e você observará que vai gerar uma resposta l #import necessary packages import flask from flask import request, jsonify - + #create the Flask application app = flask.Flask(__name__) app.config["DEBUG"] = True - + #create a list of books books = [ {'id': 0, @@ -413,18 +413,18 @@ Execute o mesmo prompt novamente e você observará que vai gerar uma resposta l 'first_sentence': 'to wound the autumnal city.', 'published': '1975'} ] - + #create an endpoint for the API @app.route('/', methods=['GET']) def home(): return '''

Books API

A prototype API for retrieving books.

''' - + #create an endpoint to return all books @app.route('/api/v1/resources/books/all', methods=['GET']) def api_all(): return jsonify(books) - + #create an endpoint to return a single book @app.route('/api/v1/resources/books', methods=['GET']) def api_id(): @@ -435,20 +435,20 @@ Execute o mesmo prompt novamente e você observará que vai gerar uma resposta l id = int(request.args['id']) else: return "Error: No id field provided. Please specify an id." - + #create an empty list for our results results = [] - + #loop through the data and match results that fit the requested ID #IDs are unique, but other fields might return many results for book in books: if book['id'] == id: results.append(book) - + #use the jsonify function from Flask to convert our list of #Python dictionaries to the JSON format return jsonify(results) - + app.run() ``` @@ -468,34 +468,34 @@ Temperatura é um valor entre 0 e 1, onde 0 é o mais determinístico e 1 é o m #import necessary libraries import flask from flask import request, jsonify - + #create an instance of the Flask class app = flask.Flask(__name__) - + #create an endpoint for the API @app.route('/api/v1/endpoint', methods=['GET']) def api_endpoint(): #get the request data data = request.get_json() - + #process the data result = process_data(data) - + #return the result return jsonify(result) - + #function to process the data def process_data(data): #process the data result = {'result': 'success'} - + #return the result return result - + #run the Flask app if __name__ == '__main__': app.run() - + ``` Execute o prompt novamente e você verá este resultado: @@ -504,34 +504,34 @@ Execute o prompt novamente e você verá este resultado: #import necessary libraries import flask from flask import request, jsonify - + #create a Flask app app = flask.Flask(__name__) - + #create a route for the API @app.route('/api', methods=['GET']) def api_call(): #get the data from the request data = request.get_json() - + #process the data result = process_data(data) - + #return the result return jsonify(result) - + #define the process_data function def process_data(data): #do something with the data result = data + 1 - + #return the result return result - + #run the app if __name__ == '__main__': app.run() - + ``` Há apenas uma pequena diferença entre essas duas saídas. Vamos fazer o contrário desta vez, vamos definir a temperatura para 0.9: @@ -540,34 +540,34 @@ Há apenas uma pequena diferença entre essas duas saídas. Vamos fazer o contr # Import necessary libraries import flask from flask import request, jsonify - + # Create a Flask app app = flask.Flask(__name__) - + # Create a route for the API @app.route('/api', methods=['GET']) def api_call(): # Get the data from the request data = request.args - + # Process the data result = process_data(data) - + # Return the result return jsonify(result) - + # Function to process the data def process_data(data): # Do something with the data result = data + 1 - + # Return the result return result - + # Run the app if __name__ == '__main__': app.run() - + ``` e a segunda tentativa em 0,9 como valor de temperatura: @@ -575,25 +575,25 @@ e a segunda tentativa em 0,9 como valor de temperatura: ```python import flask from flask import request, jsonify - + # create the Flask app app = flask.Flask(__name__) app.config['DEBUG'] = True - + # create some test data books = [ {'id': 0, 'title': 'A Fire Upon The Deep', 'author': 'Vernor Vinge', 'first_sentence': 'The coldsleep itself was dreamless.', 'year_published': '1992'}, {'id': 1, 'title': 'The Ones Who Walk Away From Omelas', 'author': 'Ursula K. Le Guin', 'first_sentence': 'With a clamor of bells that set the swallows soaring, the Festival of Summer came to the city Omelas, bright-towered by the sea.', 'published': '1973'}, {'id': 2, 'title': 'Dhalgren', 'author': 'Samuel R. Delany', 'first_sentence': 'to wound the autumnal city.', 'published': '1975'} ] - + # create an endpoint @app.route('/', methods=['GET']) def home(): return '''

Welcome to our book API!

''' - + @app.route('/api/v1/resources/books - + ``` Como você pode ver, os resultados não poderiam ser mais variados. @@ -620,14 +620,14 @@ Aqui está o código em Python que mostra como criar uma API simples usando o Fl ```python from flask import Flask, request - + app = Flask(__name__) - + @app.route('/') def hello(): name = request.args.get('name', 'World') return f'Hello, {name}!' - + if __name__ == '__main__': app.run() ``` @@ -661,4 +661,4 @@ Você acabou de usar a técnica de auto-aperfeiçoamento na atribuição. Pegue Após concluir esta lição, confira nossa [coleção de aprendizado de IA generativa](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) para continuar a aprimorar seus conhecimentos sobre IA generativa! -Agora, vamos para a Lição 6, onde aplicaremos nosso conhecimento de Engenharia de Prompt, [construindo aplicativos de geração de texto](../../../06-text-generation-apps/translations/pt-br/README.md?WT.mc_id=academic-105485-koreyst) \ No newline at end of file +Agora, vamos para a Lição 6, onde aplicaremos nosso conhecimento de Engenharia de Prompt, [construindo aplicativos de geração de texto](../../../06-text-generation-apps/translations/pt-br/README.md?WT.mc_id=academic-105485-koreyst) diff --git a/06-text-generation-apps/README.md b/06-text-generation-apps/README.md index c3ab20f78..56b46401f 100644 --- a/06-text-generation-apps/README.md +++ b/06-text-generation-apps/README.md @@ -1,8 +1,8 @@ # Building Text Generation Applications -[![Building Text Generation Applications](./images/06-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/5jKHzY6-4s8?WT.mc_id=academic-105485-koreyst) +[![Building Text Generation Applications](./images/06-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=bf3f3528-9871-4628-8616-b4b03cb23dcd?WT.mc_id=academic-105485-koreyst) -> *(Click the image above to view video of this lesson)* +> _(Click the image above to view video of this lesson)_ You've seen so far through this curriculum that there are core concepts like prompts and even a whole discipline called "prompt engineering". Many tools you can interact with like ChatGPT, Office 365, Microsoft Power Platform and more, support you using prompts to accomplish something. @@ -76,7 +76,7 @@ Let's see how we can build our first app, what libraries we need, how much is re ### Install openai -There are many libraries out there for interacting with OpenAI or Azure OpenAI. It's possible to use numerous programming languages as well like C#, Python, JavaScript, Java and more. We've chosen to use the `openai` Python library, so we'll use `pip` to install it. +There are many libraries out there for interacting with OpenAI or Azure OpenAI. It's possible to use numerous programming languages as well like C#, Python, JavaScript, Java and more. We've chosen to use the `openai` Python library, so we'll use `pip` to install it. ```bash pip install openai @@ -107,7 +107,7 @@ Now that you have this information copied, let's instruct the libraries to use i > It's worth separating your API key from your code. You can do so by using environment variables. > > - Set the environment variable `OPENAI_API_KEY` to your API key. -> `export OPENAI_API_KEY='sk-...'` +> `export OPENAI_API_KEY='sk-...'` ### Setup configuration Azure @@ -125,10 +125,9 @@ Above we're setting the following: - `api_type` to `azure`. This tells the library to use Azure Open AI and not OpenAI. - `api_key`, this is your API key found in the Azure Portal. - `api_version`, this is the version of the API you want to use. At the time of writing, the latest version is `2023-05-15`. -- `api_base`, this is the endpoint of the API. You can find it in the Azure Portal next to your API key. +- `api_base`, this is the endpoint of the API. You can find it in the Azure Portal next to your API key. -> [!NOTE] -> `os.getenv` is a function that reads environment variables. You can use it to read environment variables like `OPENAI_API_KEY` and `API_BASE`. Set these environment variables in your terminal or by using a library like `dotenv`. +> [!NOTE] > `os.getenv` is a function that reads environment variables. You can use it to read environment variables like `OPENAI_API_KEY` and `API_BASE`. Set these environment variables in your terminal or by using a library like `dotenv`. ## Generate text @@ -164,51 +163,51 @@ Now that we learned how to set up and configure openai, it's time to build your 1. Create a virtual environment and install openai: - ```bash - python -m venv venv - source venv/bin/activate - pip install openai - ``` + ```bash + python -m venv venv + source venv/bin/activate + pip install openai + ``` + + > [!NOTE] + > If you're using Windows type `venv\Scripts\activate` instead of `source venv/bin/activate`. - > [!NOTE] - > If you're using Windows type `venv\Scripts\activate` instead of `source venv/bin/activate`. + > [!NOTE] + > Locate your Azure Open AI key by going to [https://portal.azure.com/](https://portal.azure.com/?WT.mc_id=academic-105485-koreyst) and search for `Open AI` and select the `Open AI resource` and then select `Keys and Endpoint` and copy the `Key 1` value. - > [!NOTE] - > Locate your Azure Open AI key by going to [https://portal.azure.com/](https://portal.azure.com/?WT.mc_id=academic-105485-koreyst) and search for `Open AI` and select the `Open AI resource` and then select `Keys and Endpoint` and copy the `Key 1` value. +1. Create an _app.py_ file and give it the following code: -1. Create an *app.py* file and give it the following code: + ```python + import openai - ```python - import openai + openai.api_key = "" - openai.api_key = "" + openai.api_type = 'azure' + openai.api_version = '2023-05-15' + openai.api_base = "" + deployment_name = "" - openai.api_type = 'azure' - openai.api_version = '2023-05-15' - openai.api_base = "" - deployment_name = "" + # add your completion code + prompt = "Complete the following: Once upon a time there was a" + messages = [{"role": "user", "content": prompt}] - # add your completion code - prompt = "Complete the following: Once upon a time there was a" - messages = [{"role": "user", "content": prompt}] + # make completion + completion = client.chat.completions.create(model=deployment, messages=messages) - # make completion - completion = client.chat.completions.create(model=deployment, messages=messages) - - # print response - print(completion.choices[0].message.content) - ``` + # print response + print(completion.choices[0].message.content) + ``` - > [!NOTE] - > If you're using Azure Open AI, you need to set the `api_type` to `azure` and set the `api_key` to your Azure Open AI key. + > [!NOTE] + > If you're using Azure Open AI, you need to set the `api_type` to `azure` and set the `api_key` to your Azure Open AI key. - You should see an output like the following: + You should see an output like the following: - ```output - very unhappy _____. + ```output + very unhappy _____. - Once upon a time there was a very unhappy mermaid. - ``` + Once upon a time there was a very unhappy mermaid. + ``` ## Different types of prompts, for different things @@ -231,8 +230,8 @@ You could write a prompt like so: Given the above prompt, you might get a response similar to: ```output -1. Roasted Chicken and Vegetables: -Ingredients: +1. Roasted Chicken and Vegetables: +Ingredients: - 4 chicken thighs - 2 potatoes, cut into cubes - 2 carrots, cut into cubes @@ -242,8 +241,8 @@ Ingredients: - 1 teaspoon dried oregano - Salt and pepper, to taste -2. Chicken and Potato Stew: -Ingredients: +2. Chicken and Potato Stew: +Ingredients: - 2 tablespoons olive oil - 1 onion, diced - 2 cloves garlic, minced @@ -255,8 +254,8 @@ Ingredients: - 1 cup chicken broth - Salt and pepper, to taste -3. Chicken and Potato Bake: -Ingredients: +3. Chicken and Potato Bake: +Ingredients: - 2 tablespoons olive oil - 2 chicken breasts, cut into cubes - 2 potatoes, cut into cubes @@ -268,8 +267,8 @@ Ingredients: - 1 cup chicken broth - Salt and pepper, to taste -4. Chicken and Potato Soup: -Ingredients: +4. Chicken and Potato Soup: +Ingredients: - 2 tablespoons olive oil - 1 onion, diced - 2 cloves garlic, minced @@ -281,8 +280,8 @@ Ingredients: - 4 cups chicken broth - Salt and pepper, to taste -5. Chicken and Potato Hash: -Ingredients: +5. Chicken and Potato Hash: +Ingredients: - 2 tablespoons olive oil - 2 chicken breasts, cut into cubes - 2 potatoes, cut into cubes @@ -304,8 +303,8 @@ For the above cases, let's add an additional prompt: Now you have a new result, namely: ```output -1. Roasted Chicken and Vegetables: -Ingredients: +1. Roasted Chicken and Vegetables: +Ingredients: - 4 chicken thighs - 2 potatoes, cut into cubes - 2 carrots, cut into cubes @@ -314,8 +313,8 @@ Ingredients: - 1 teaspoon dried oregano - Salt and pepper, to taste -2. Chicken and Potato Stew: -Ingredients: +2. Chicken and Potato Stew: +Ingredients: - 2 tablespoons olive oil - 1 onion, diced - 2 chicken breasts, cut into cubes @@ -326,8 +325,8 @@ Ingredients: - 1 cup chicken broth - Salt and pepper, to taste -3. Chicken and Potato Bake: -Ingredients: +3. Chicken and Potato Bake: +Ingredients: - 2 tablespoons olive oil - 2 chicken breasts, cut into cubes - 2 potatoes, cut into cubes @@ -338,8 +337,8 @@ Ingredients: - 1 cup chicken broth - Salt and pepper, to taste -4. Chicken and Potato Soup: -Ingredients: +4. Chicken and Potato Soup: +Ingredients: - 2 tablespoons olive oil - 1 onion, diced - 2 chicken breasts, cut into cubes @@ -350,8 +349,8 @@ Ingredients: - 4 cups chicken broth - Salt and pepper, to taste -5. Chicken and Potato Hash: -Ingredients: +5. Chicken and Potato Hash: +Ingredients: - 2 tablespoons olive oil - 2 chicken breasts, cut into cubes - 2 potatoes, cut into cubes @@ -359,7 +358,7 @@ Ingredients: - 1 onion, diced - 1 teaspoon dried oregano -Shopping List: +Shopping List: - Olive oil - Onion - Thyme @@ -374,50 +373,50 @@ That's your five recipes, with no garlic mentioned and you also have a shopping Now that we have played out a scenario, let's write code to match the demonstrated scenario. To do so, follow these steps: -1. Use the existing *app.py* file as a starting point +1. Use the existing _app.py_ file as a starting point 1. Locate the `prompt` variable and change its code to the following: - ```python - prompt = "Show me 5 recipes for a dish with the following ingredients: chicken, potatoes, and carrots. Per recipe, list all the ingredients used" - ``` + ```python + prompt = "Show me 5 recipes for a dish with the following ingredients: chicken, potatoes, and carrots. Per recipe, list all the ingredients used" + ``` + + If you now run the code, you should see an output similar to: + + ```output + -Chicken Stew with Potatoes and Carrots: 3 tablespoons oil, 1 onion, chopped, 2 cloves garlic, minced, 1 carrot, peeled and chopped, 1 potato, peeled and chopped, 1 bay leaf, 1 thyme sprig, 1/2 teaspoon salt, 1/4 teaspoon black pepper, 1 1/2 cups chicken broth, 1/2 cup dry white wine, 2 tablespoons chopped fresh parsley, 2 tablespoons unsalted butter, 1 1/2 pounds boneless, skinless chicken thighs, cut into 1-inch pieces + -Oven-Roasted Chicken with Potatoes and Carrots: 3 tablespoons extra-virgin olive oil, 1 tablespoon Dijon mustard, 1 tablespoon chopped fresh rosemary, 1 tablespoon chopped fresh thyme, 4 cloves garlic, minced, 1 1/2 pounds small red potatoes, quartered, 1 1/2 pounds carrots, quartered lengthwise, 1/2 teaspoon salt, 1/4 teaspoon black pepper, 1 (4-pound) whole chicken + -Chicken, Potato, and Carrot Casserole: cooking spray, 1 large onion, chopped, 2 cloves garlic, minced, 1 carrot, peeled and shredded, 1 potato, peeled and shredded, 1/2 teaspoon dried thyme leaves, 1/4 teaspoon salt, 1/4 teaspoon black pepper, 2 cups fat-free, low-sodium chicken broth, 1 cup frozen peas, 1/4 cup all-purpose flour, 1 cup 2% reduced-fat milk, 1/4 cup grated Parmesan cheese - If you now run the code, you should see an output similar to: + -One Pot Chicken and Potato Dinner: 2 tablespoons olive oil, 1 pound boneless, skinless chicken thighs, cut into 1-inch pieces, 1 large onion, chopped, 3 cloves garlic, minced, 1 carrot, peeled and chopped, 1 potato, peeled and chopped, 1 bay leaf, 1 thyme sprig, 1/2 teaspoon salt, 1/4 teaspoon black pepper, 2 cups chicken broth, 1/2 cup dry white wine - ```output - -Chicken Stew with Potatoes and Carrots: 3 tablespoons oil, 1 onion, chopped, 2 cloves garlic, minced, 1 carrot, peeled and chopped, 1 potato, peeled and chopped, 1 bay leaf, 1 thyme sprig, 1/2 teaspoon salt, 1/4 teaspoon black pepper, 1 1/2 cups chicken broth, 1/2 cup dry white wine, 2 tablespoons chopped fresh parsley, 2 tablespoons unsalted butter, 1 1/2 pounds boneless, skinless chicken thighs, cut into 1-inch pieces - -Oven-Roasted Chicken with Potatoes and Carrots: 3 tablespoons extra-virgin olive oil, 1 tablespoon Dijon mustard, 1 tablespoon chopped fresh rosemary, 1 tablespoon chopped fresh thyme, 4 cloves garlic, minced, 1 1/2 pounds small red potatoes, quartered, 1 1/2 pounds carrots, quartered lengthwise, 1/2 teaspoon salt, 1/4 teaspoon black pepper, 1 (4-pound) whole chicken - -Chicken, Potato, and Carrot Casserole: cooking spray, 1 large onion, chopped, 2 cloves garlic, minced, 1 carrot, peeled and shredded, 1 potato, peeled and shredded, 1/2 teaspoon dried thyme leaves, 1/4 teaspoon salt, 1/4 teaspoon black pepper, 2 cups fat-free, low-sodium chicken broth, 1 cup frozen peas, 1/4 cup all-purpose flour, 1 cup 2% reduced-fat milk, 1/4 cup grated Parmesan cheese - - -One Pot Chicken and Potato Dinner: 2 tablespoons olive oil, 1 pound boneless, skinless chicken thighs, cut into 1-inch pieces, 1 large onion, chopped, 3 cloves garlic, minced, 1 carrot, peeled and chopped, 1 potato, peeled and chopped, 1 bay leaf, 1 thyme sprig, 1/2 teaspoon salt, 1/4 teaspoon black pepper, 2 cups chicken broth, 1/2 cup dry white wine - - -Chicken, Potato, and Carrot Curry: 1 tablespoon vegetable oil, 1 large onion, chopped, 2 cloves garlic, minced, 1 carrot, peeled and chopped, 1 potato, peeled and chopped, 1 teaspoon ground coriander, 1 teaspoon ground cumin, 1/2 teaspoon ground turmeric, 1/2 teaspoon ground ginger, 1/4 teaspoon cayenne pepper, 2 cups chicken broth, 1/2 cup dry white wine, 1 (15-ounce) can chickpeas, drained and rinsed, 1/2 cup raisins, 1/2 cup chopped fresh cilantro - ``` + -Chicken, Potato, and Carrot Curry: 1 tablespoon vegetable oil, 1 large onion, chopped, 2 cloves garlic, minced, 1 carrot, peeled and chopped, 1 potato, peeled and chopped, 1 teaspoon ground coriander, 1 teaspoon ground cumin, 1/2 teaspoon ground turmeric, 1/2 teaspoon ground ginger, 1/4 teaspoon cayenne pepper, 2 cups chicken broth, 1/2 cup dry white wine, 1 (15-ounce) can chickpeas, drained and rinsed, 1/2 cup raisins, 1/2 cup chopped fresh cilantro + ``` - > NOTE, your LLM is nondeterministic, so you might get different results every time you run the program. + > NOTE, your LLM is nondeterministic, so you might get different results every time you run the program. - Great, let's see how we can improve things. To improve things, we want to make sure the code is flexible, so ingredients and number of recipes can be improved and changed. + Great, let's see how we can improve things. To improve things, we want to make sure the code is flexible, so ingredients and number of recipes can be improved and changed. 1. Let's change the code in the following way: - ```python - no_recipes = input("No of recipes (for example, 5: ") + ```python + no_recipes = input("No of recipes (for example, 5: ") - ingredients = input("List of ingredients (for example, chicken, potatoes, and carrots: ") - - # interpolate the number of recipes into the prompt an ingredients - prompt = f"Show me {no_recipes} recipes for a dish with the following ingredients: {ingredients}. Per recipe, list all the ingredients used" - ``` + ingredients = input("List of ingredients (for example, chicken, potatoes, and carrots: ") - Taking the code for a test run, could look like this: + # interpolate the number of recipes into the prompt an ingredients + prompt = f"Show me {no_recipes} recipes for a dish with the following ingredients: {ingredients}. Per recipe, list all the ingredients used" + ``` - ```output - No of recipes (for example, 5: 3 - List of ingredients (for example, chicken, potatoes, and carrots: milk,strawberries + Taking the code for a test run, could look like this: - -Strawberry milk shake: milk, strawberries, sugar, vanilla extract, ice cubes - -Strawberry shortcake: milk, flour, baking powder, sugar, salt, unsalted butter, strawberries, whipped cream - -Strawberry milk: milk, strawberries, sugar, vanilla extract - ``` + ```output + No of recipes (for example, 5: 3 + List of ingredients (for example, chicken, potatoes, and carrots: milk,strawberries + + -Strawberry milk shake: milk, strawberries, sugar, vanilla extract, ice cubes + -Strawberry shortcake: milk, flour, baking powder, sugar, salt, unsalted butter, strawberries, whipped cream + -Strawberry milk: milk, strawberries, sugar, vanilla extract + ``` ### Improve by adding filter and shopping list @@ -427,130 +426,129 @@ To further improve it, we want to add the following: - **Filter out ingredients**. We want to be able to filter out ingredients we don't like or are allergic to. To accomplish this change, we can edit our existing prompt and add a filter condition to the end of it like so: - ```python - filter = input("Filter (for example, vegetarian, vegan, or gluten-free: ") - - prompt = f"Show me {no_recipes} recipes for a dish with the following ingredients: {ingredients}. Per recipe, list all the ingredients used, no {filter}" - ``` - - Above, we add `{filter}` to the end of the prompt and we also capture the filter value from the user. - - An example input of running the program can now look like so: - - ```output - No of recipes (for example, 5: 3 - List of ingredients (for example, chicken, potatoes, and carrots: onion,milk - Filter (for example, vegetarian, vegan, or gluten-free: no milk - - 1. French Onion Soup - - Ingredients: - - -1 large onion, sliced - -3 cups beef broth - -1 cup milk - -6 slices french bread - -1/4 cup shredded Parmesan cheese - -1 tablespoon butter - -1 teaspoon dried thyme - -1/4 teaspoon salt - -1/4 teaspoon black pepper - - Instructions: - - 1. In a large pot, sauté onions in butter until golden brown. - 2. Add beef broth, milk, thyme, salt, and pepper. Bring to a boil. - 3. Reduce heat and simmer for 10 minutes. - 4. Place french bread slices on soup bowls. - 5. Ladle soup over bread. - 6. Sprinkle with Parmesan cheese. - - 2. Onion and Potato Soup - - Ingredients: - - -1 large onion, chopped - -2 cups potatoes, diced - -3 cups vegetable broth - -1 cup milk - -1/4 teaspoon black pepper - - Instructions: - - 1. In a large pot, sauté onions in butter until golden brown. - 2. Add potatoes, vegetable broth, milk, and pepper. Bring to a boil. - 3. Reduce heat and simmer for 10 minutes. - 4. Serve hot. - - 3. Creamy Onion Soup - - Ingredients: - - -1 large onion, chopped - -3 cups vegetable broth - -1 cup milk - -1/4 teaspoon black pepper - -1/4 cup all-purpose flour - -1/2 cup shredded Parmesan cheese - - Instructions: - - 1. In a large pot, sauté onions in butter until golden brown. - 2. Add vegetable broth, milk, and pepper. Bring to a boil. - 3. Reduce heat and simmer for 10 minutes. - 4. In a small bowl, whisk together flour and Parmesan cheese until smooth. - 5. Add to soup and simmer for an additional 5 minutes, or until soup has thickened. - ``` - - As you can see, any recipes with milk in it has been filtered out. But, if you're lactose intolerant, you might want to filter out recipes with cheese in them as well, so there's a need to be clear. + ```python + filter = input("Filter (for example, vegetarian, vegan, or gluten-free: ") + + prompt = f"Show me {no_recipes} recipes for a dish with the following ingredients: {ingredients}. Per recipe, list all the ingredients used, no {filter}" + ``` + + Above, we add `{filter}` to the end of the prompt and we also capture the filter value from the user. + + An example input of running the program can now look like so: + + ```output + No of recipes (for example, 5: 3 + List of ingredients (for example, chicken, potatoes, and carrots: onion,milk + Filter (for example, vegetarian, vegan, or gluten-free: no milk + + 1. French Onion Soup + + Ingredients: + + -1 large onion, sliced + -3 cups beef broth + -1 cup milk + -6 slices french bread + -1/4 cup shredded Parmesan cheese + -1 tablespoon butter + -1 teaspoon dried thyme + -1/4 teaspoon salt + -1/4 teaspoon black pepper + + Instructions: + + 1. In a large pot, sauté onions in butter until golden brown. + 2. Add beef broth, milk, thyme, salt, and pepper. Bring to a boil. + 3. Reduce heat and simmer for 10 minutes. + 4. Place french bread slices on soup bowls. + 5. Ladle soup over bread. + 6. Sprinkle with Parmesan cheese. + 2. Onion and Potato Soup + + Ingredients: + + -1 large onion, chopped + -2 cups potatoes, diced + -3 cups vegetable broth + -1 cup milk + -1/4 teaspoon black pepper + + Instructions: + + 1. In a large pot, sauté onions in butter until golden brown. + 2. Add potatoes, vegetable broth, milk, and pepper. Bring to a boil. + 3. Reduce heat and simmer for 10 minutes. + 4. Serve hot. + + 3. Creamy Onion Soup + + Ingredients: + + -1 large onion, chopped + -3 cups vegetable broth + -1 cup milk + -1/4 teaspoon black pepper + -1/4 cup all-purpose flour + -1/2 cup shredded Parmesan cheese + + Instructions: + + 1. In a large pot, sauté onions in butter until golden brown. + 2. Add vegetable broth, milk, and pepper. Bring to a boil. + 3. Reduce heat and simmer for 10 minutes. + 4. In a small bowl, whisk together flour and Parmesan cheese until smooth. + 5. Add to soup and simmer for an additional 5 minutes, or until soup has thickened. + ``` + + As you can see, any recipes with milk in it has been filtered out. But, if you're lactose intolerant, you might want to filter out recipes with cheese in them as well, so there's a need to be clear. - **Produce a shopping list**. We want to produce a shopping list, considering what we already have at home. - For this functionality, we could either try to solve everything in one prompt or we could split it up into two prompts. Let's try the latter approach. Here we're suggesting adding an additional prompt, but for that to work, we need to add the result of the former prompt as context to the latter prompt. + For this functionality, we could either try to solve everything in one prompt or we could split it up into two prompts. Let's try the latter approach. Here we're suggesting adding an additional prompt, but for that to work, we need to add the result of the former prompt as context to the latter prompt. + + Locate the part in the code that prints out the result from the first prompt and add the following code below: - Locate the part in the code that prints out the result from the first prompt and add the following code below: + ```python + old_prompt_result = completion.choices[0].message.content + prompt = "Produce a shopping list for the generated recipes and please don't include ingredients that I already have." - ```python - old_prompt_result = completion.choices[0].message.content - prompt = "Produce a shopping list for the generated recipes and please don't include ingredients that I already have." - - new_prompt = f"{old_prompt_result} {prompt}" - messages = [{"role": "user", "content": new_prompt}] - completion = openai.Completion.create(engine=deployment_name, messages=messages, max_tokens=1200) - - # print response - print("Shopping list:") - print(completion.choices[0].message.content) - ``` + new_prompt = f"{old_prompt_result} {prompt}" + messages = [{"role": "user", "content": new_prompt}] + completion = openai.Completion.create(engine=deployment_name, messages=messages, max_tokens=1200) - Note the following: + # print response + print("Shopping list:") + print(completion.choices[0].message.content) + ``` - 1. We're constructing a new prompt by adding the result from the first prompt to the new prompt: + Note the following: - ```python - new_prompt = f"{old_prompt_result} {prompt}" - ``` + 1. We're constructing a new prompt by adding the result from the first prompt to the new prompt: - 1. We make a new request, but also considering the number of tokens we asked for in the first prompt, so this time we say `max_tokens` is 1200. + ```python + new_prompt = f"{old_prompt_result} {prompt}" + ``` - ```python - completion = openai.Completion.create(engine=deployment_name, prompt=new_prompt, max_tokens=1200) - ``` + 1. We make a new request, but also considering the number of tokens we asked for in the first prompt, so this time we say `max_tokens` is 1200. - Taking this code for a spin, we now arrive at the following output: + ```python + completion = openai.Completion.create(engine=deployment_name, prompt=new_prompt, max_tokens=1200) + ``` - ```output - No of recipes (for example, 5: 2 - List of ingredients (for example, chicken, potatoes, and carrots: apple,flour - Filter (for example, vegetarian, vegan, or gluten-free: sugar + Taking this code for a spin, we now arrive at the following output: - - -Apple and flour pancakes: 1 cup flour, 1/2 tsp baking powder, 1/2 tsp baking soda, 1/4 tsp salt, 1 tbsp sugar, 1 egg, 1 cup buttermilk or sour milk, 1/4 cup melted butter, 1 Granny Smith apple, peeled and grated - -Apple fritters: 1-1/2 cups flour, 1 tsp baking powder, 1/4 tsp salt, 1/4 tsp baking soda, 1/4 tsp nutmeg, 1/4 tsp cinnamon, 1/4 tsp allspice, 1/4 cup sugar, 1/4 cup vegetable shortening, 1/4 cup milk, 1 egg, 2 cups shredded, peeled apples - Shopping list: - -Flour, baking powder, baking soda, salt, sugar, egg, buttermilk, butter, apple, nutmeg, cinnamon, allspice - ``` + ```output + No of recipes (for example, 5: 2 + List of ingredients (for example, chicken, potatoes, and carrots: apple,flour + Filter (for example, vegetarian, vegan, or gluten-free: sugar + + + -Apple and flour pancakes: 1 cup flour, 1/2 tsp baking powder, 1/2 tsp baking soda, 1/4 tsp salt, 1 tbsp sugar, 1 egg, 1 cup buttermilk or sour milk, 1/4 cup melted butter, 1 Granny Smith apple, peeled and grated + -Apple fritters: 1-1/2 cups flour, 1 tsp baking powder, 1/4 tsp salt, 1/4 tsp baking soda, 1/4 tsp nutmeg, 1/4 tsp cinnamon, 1/4 tsp allspice, 1/4 cup sugar, 1/4 cup vegetable shortening, 1/4 cup milk, 1 egg, 2 cups shredded, peeled apples + Shopping list: + -Flour, baking powder, baking soda, salt, sugar, egg, buttermilk, butter, apple, nutmeg, cinnamon, allspice + ``` ## Improve your setup @@ -558,47 +556,47 @@ What we have so far is code that works, but there are some tweaks we should be d - **Separate secrets from code**, like the API key. Secrets do not belong in code and should be stored in a secure location. To separate secrets from code, we can use environment variables and libraries like `python-dotenv` to load them from a file. Here's how that would look like in code: - 1. Create a `.env` file with the following content: + 1. Create a `.env` file with the following content: - ```bash - OPENAI_API_KEY=sk-... - ``` + ```bash + OPENAI_API_KEY=sk-... + ``` - > Note, for Azure, you need to set the following environment variables: + > Note, for Azure, you need to set the following environment variables: - ```bash - OPENAI_API_TYPE=azure - OPENAI_API_VERSION=2023-05-15 - OPENAI_API_BASE= - ``` + ```bash + OPENAI_API_TYPE=azure + OPENAI_API_VERSION=2023-05-15 + OPENAI_API_BASE= + ``` - In code, you would load the environment variables like so: + In code, you would load the environment variables like so: - ```python - from dotenv import load_dotenv + ```python + from dotenv import load_dotenv - load_dotenv() + load_dotenv() - openai.api_key = os.environ["OPENAI_API_KEY"] - ``` + openai.api_key = os.environ["OPENAI_API_KEY"] + ``` - **A word on token length**. We should consider how many tokens we need to generate the text we want. Tokens cost money, so where possible, we should try to be economical with the number of tokens we use. For example, can we phrase the prompt so that we can use less tokens? - To change the tokens used, you can use the `max_tokens` parameter. For example, if you want to use 100 tokens, you would do: + To change the tokens used, you can use the `max_tokens` parameter. For example, if you want to use 100 tokens, you would do: - ```python - completion = client.chat.completions.create(model=deployment, messages=messages, max_tokens=100) - ``` + ```python + completion = client.chat.completions.create(model=deployment, messages=messages, max_tokens=100) + ``` - **Experimenting with temperature**. Temperature is something we haven't mentioned so far but is an important context for how our program performs. The higher the temperature value the more random the output will be. Conversely the lower the temperature value the more predictable the output will be. Consider whether you want variation in your output or not. - To alter the temperature, you can use the `temperature` parameter. For example, if you want to use a temperature of 0.5, you would do: + To alter the temperature, you can use the `temperature` parameter. For example, if you want to use a temperature of 0.5, you would do: - ```python - completion = client.chat.completions.create(model=deployment, messages=messages, temperature=0.5) - ``` + ```python + completion = client.chat.completions.create(model=deployment, messages=messages, temperature=0.5) + ``` - > Note, the closer to 1.0, the more varied the output. + > Note, the closer to 1.0, the more varied the output. ## Assignment @@ -646,7 +644,6 @@ What does the concept temperature do? 1. It controls how big the response is. 1. It controls how many tokens are used. - ## 🚀 Challenge When working on the assignment, try to vary the temperature, try set it to 0, 0.5, and 1. Remember that 0 is the least varied and 1 is the most, what value works best for your app? diff --git a/06-text-generation-apps/translations/cn/README.md b/06-text-generation-apps/translations/cn/README.md index d162d9035..1b79515d9 100644 --- a/06-text-generation-apps/translations/cn/README.md +++ b/06-text-generation-apps/translations/cn/README.md @@ -1,8 +1,8 @@ # 第六章:创建文本生成应用 -[![Building Text Generation Applications](../../images/06-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/5jKHzY6-4s8?WT.mc_id=academic-105485-koreyst) +[![Building Text Generation Applications](../../images/06-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=bf3f3528-9871-4628-8616-b4b03cb23dcd?WT.mc_id=academic-105485-koreyst) -> *(点击该图片看本章导学视频)* +> _(点击该图片看本章导学视频)_ 到目前为止,您已经通过本系列课程看到了如提示之类的核心概念,甚至是称为“提示工程”的整个学科。 您可以与通过使用如 ChatGPT、Office 365、Microsoft Power Platform 等工具结合提示来完成某些工作任务。 @@ -24,7 +24,6 @@ - 使用 openai 构建文本生成应用程序。 - 配置您的应用程序以使用更多或更少的 tokens,并更改 temperature,以获得不同的输出。 - ## 什么是文本生成应用 一般来说当您构建应用程序时,它具有某种界面,如下所示: @@ -53,7 +52,6 @@ - **协同助手** LLMs 擅长总结文本、从文本中获取见解、生成简历等文本等。 - **代码助手** 根据您使用的编程语言模型,您可以构建一个代码助手来帮助您编写代码。 例如,您可以使用 GitHub Copilot 和 ChatGPT 等产品来帮助您编写代码。 - ## 如何入门? 您需要找到一种与 LLMs 结合的方法,通常使用以下两种方法: @@ -76,11 +74,11 @@ Let's see how we can build our first app, what libraries we need, how much is required and so on. -让我们看看如何构建人生中第一个 openai 应用,我们需要哪些libraries,需要多少技能等等。 +让我们看看如何构建人生中第一个 openai 应用,我们需要哪些 libraries,需要多少技能等等。 ### 安装 openai -There are many libraries out there for interacting with OpenAI or Azure OpenAI. It's possible to use numerous programming languages as well like C#, Python, JavaScript, Java and more. We've chosen to use the `openai` Python library, so we'll use `pip` to install it. +There are many libraries out there for interacting with OpenAI or Azure OpenAI. It's possible to use numerous programming languages as well like C#, Python, JavaScript, Java and more. We've chosen to use the `openai` Python library, so we'll use `pip` to install it. 有许多 libraries 可用于与 OpenAI 或 Azure OpenAI 交互。 还可以使用不同的编程语言,如 C#、Python、JavaScript、Java 等。 我们选择使用 `openai` Python 库,通过 `pip` 来安装它。 @@ -112,8 +110,8 @@ pip install openai > [!注意] > 通过设置环境变量将 API Key 与代码分开是很重要的 > -> - 为您的 API key 中设置 `OPENAI_API_KEY` -> - `export OPENAI_API_KEY='sk-...'` +> - 为您的 API key 中设置 `OPENAI_API_KEY` +> - `export OPENAI_API_KEY='sk-...'` ### 配置 Azure 环境 @@ -134,7 +132,7 @@ openai.api_base = os.getenv("API_BASE") - `api_base`, 这是 API 的 endpoint 。 您可以在 Azure Portal 中 API Key 下方找到它。 > [注意] -> `os.getenv` 是一个读取环境变量的函数。 您可以使用它来读取“OPENAI_API_KEY”和“API_BASE”等环境变量。 在终端中或使用“dotenv”等库设置这些环境变量。 +> `os.getenv` 是一个读取环境变量的函数。 您可以使用它来读取“OPENAI_API_KEY”和“API_BASE”等环境变量。 在终端中或使用“dotenv”等库设置这些环境变量。 ## 文字生成 @@ -168,52 +166,52 @@ print(completion.choices[0].message.content) 现在我们已经了解了如何设置和配置 openai,是时候构建您的第一个文本生成应用程序了。 请按照下列步骤操作: -1. 创建虚拟环境并安装openai: +1. 创建虚拟环境并安装 openai: - ```bash - python -m venv venv - source venv/bin/activate - pip install openai - ``` + ```bash + python -m venv venv + source venv/bin/activate + pip install openai + ``` - > [!注意] - > 如果您使用的是 Windows,请输入 `venv\Scripts\activate` 而不是 `source venv/bin/activate`。 + > [!注意] + > 如果您使用的是 Windows,请输入 `venv\Scripts\activate` 而不是 `source venv/bin/activate`。 - > [!注意] - > 转至 [https://portal.azure.com/](https://portal.azure.com/?WT.mc_id=academic-105485-koreyst) 找到您的 Azure Open AI Key ,在资源中搜索“Open AI” ` 并选择“打开 AI 资源”,然后选择 `Keys and Endpoint` 并复制 `Key 1` 值。 + > [!注意] + > 转至 [https://portal.azure.com/](https://portal.azure.com/?WT.mc_id=academic-105485-koreyst) 找到您的 Azure Open AI Key ,在资源中搜索“Open AI” `并选择“打开 AI 资源”,然后选择`Keys and Endpoint`并复制`Key 1` 值。 -2. 创建 *app.py* 文件并添加以下代码: +2. 创建 _app.py_ 文件并添加以下代码: - ```python - import openai + ```python + import openai - openai.api_key = "您的 openai key 或 Azure OpenAI key" + openai.api_key = "您的 openai key 或 Azure OpenAI key" - openai.api_type = 'azure' - openai.api_version = '2023-05-15' - openai.api_base = "您的 Azure OpenAI Endpoint" - deployment_name = "部署模型的名字" + openai.api_type = 'azure' + openai.api_version = '2023-05-15' + openai.api_base = "您的 Azure OpenAI Endpoint" + deployment_name = "部署模型的名字" - # add your completion code - prompt = "Complete the following: Once upon a time there was a" + # add your completion code + prompt = "Complete the following: Once upon a time there was a" - # make completion - completion = openai.Completion.create(engine= deployment_name, model="davinci-002", prompt=prompt) - - # print response - print(completion.choices[0].text) - ``` + # make completion + completion = openai.Completion.create(engine= deployment_name, model="davinci-002", prompt=prompt) - > [!注意] - > 如果您使用的是 Azure Open AI,则需要将 `api_type` 设置为 `azure`,并将 `api_key` 设置为您的 Azure Open AI Key + # print response + print(completion.choices[0].text) + ``` - 您应该看到如下所示的输出结果: + > [!注意] + > 如果您使用的是 Azure Open AI,则需要将 `api_type` 设置为 `azure`,并将 `api_key` 设置为您的 Azure Open AI Key - ```output - very unhappy _____. + 您应该看到如下所示的输出结果: - Once upon a time there was a very unhappy mermaid. - ``` + ```output + very unhappy _____. + + Once upon a time there was a very unhappy mermaid. + ``` ## 不同类型的提示,针对不同的事情 @@ -223,8 +221,7 @@ print(completion.choices[0].message.content) - **生成一种类型的文本**。 例如,生成一首诗、测验题目等。 - **查找信息**。 您可以使用提示来查找信息,例如以下示例'What does CORS mean in web development?'。 -- **生成代码**。 您可以使用提示来生成代码,例如开发用于验证电子邮件的正则表达式,或者为什么不生成整个程序,例如web 应用? - +- **生成代码**。 您可以使用提示来生成代码,例如开发用于验证电子邮件的正则表达式,或者为什么不生成整个程序,例如 web 应用? ## 进阶学习:菜谱生成器 @@ -237,8 +234,8 @@ print(completion.choices[0].message.content) 鉴于上述提示,您可能会得到类似于以下结果: ```output -1. Roasted Chicken and Vegetables: -Ingredients: +1. Roasted Chicken and Vegetables: +Ingredients: - 4 chicken thighs - 2 potatoes, cut into cubes - 2 carrots, cut into cubes @@ -248,8 +245,8 @@ Ingredients: - 1 teaspoon dried oregano - Salt and pepper, to taste -1. Chicken and Potato Stew: -Ingredients: +1. Chicken and Potato Stew: +Ingredients: - 2 tablespoons olive oil - 1 onion, diced - 2 cloves garlic, minced @@ -261,8 +258,8 @@ Ingredients: - 1 cup chicken broth - Salt and pepper, to taste -1. Chicken and Potato Bake: -Ingredients: +1. Chicken and Potato Bake: +Ingredients: - 2 tablespoons olive oil - 2 chicken breasts, cut into cubes - 2 potatoes, cut into cubes @@ -274,8 +271,8 @@ Ingredients: - 1 cup chicken broth - Salt and pepper, to taste -1. Chicken and Potato Soup: -Ingredients: +1. Chicken and Potato Soup: +Ingredients: - 2 tablespoons olive oil - 1 onion, diced - 2 cloves garlic, minced @@ -287,8 +284,8 @@ Ingredients: - 4 cups chicken broth - Salt and pepper, to taste -1. Chicken and Potato Hash: -Ingredients: +1. Chicken and Potato Hash: +Ingredients: - 2 tablespoons olive oil - 2 chicken breasts, cut into cubes - 2 potatoes, cut into cubes @@ -310,8 +307,8 @@ Ingredients: 现在你有了一个新的结果,即: ```output -1. Roasted Chicken and Vegetables: -Ingredients: +1. Roasted Chicken and Vegetables: +Ingredients: - 4 chicken thighs - 2 potatoes, cut into cubes - 2 carrots, cut into cubes @@ -320,8 +317,8 @@ Ingredients: - 1 teaspoon dried oregano - Salt and pepper, to taste -1. Chicken and Potato Stew: -Ingredients: +1. Chicken and Potato Stew: +Ingredients: - 2 tablespoons olive oil - 1 onion, diced - 2 chicken breasts, cut into cubes @@ -332,8 +329,8 @@ Ingredients: - 1 cup chicken broth - Salt and pepper, to taste -1. Chicken and Potato Bake: -Ingredients: +1. Chicken and Potato Bake: +Ingredients: - 2 tablespoons olive oil - 2 chicken breasts, cut into cubes - 2 potatoes, cut into cubes @@ -344,8 +341,8 @@ Ingredients: - 1 cup chicken broth - Salt and pepper, to taste -1. Chicken and Potato Soup: -Ingredients: +1. Chicken and Potato Soup: +Ingredients: - 2 tablespoons olive oil - 1 onion, diced - 2 chicken breasts, cut into cubes @@ -356,8 +353,8 @@ Ingredients: - 4 cups chicken broth - Salt and pepper, to taste -1. Chicken and Potato Hash: -Ingredients: +1. Chicken and Potato Hash: +Ingredients: - 2 tablespoons olive oil - 2 chicken breasts, cut into cubes - 2 potatoes, cut into cubes @@ -365,7 +362,7 @@ Ingredients: - 1 onion, diced - 1 teaspoon dried oregano -Shopping List: +Shopping List: - Olive oil - Onion - Thyme @@ -382,54 +379,52 @@ That's your five recipes, with no garlic mentioned and you also have a shopping 根据场景,让我们编写代码来完整代码。为此,请按照下列步骤操作: -1. 使用现有的 *app.py* 文件作为起点 +1. 使用现有的 _app.py_ 文件作为起点 2. 找到 `prompt` 变量并将其代码更改为以下内容: + ```python + prompt = "Show me 5 recipes for a dish with the following ingredients: chicken, potatoes, and carrots. Per recipe, list all the ingredients used" + ``` - ```python - prompt = "Show me 5 recipes for a dish with the following ingredients: chicken, potatoes, and carrots. Per recipe, list all the ingredients used" - ``` + 运行代码,您应该会看到类似以下内容的输出: - 运行代码,您应该会看到类似以下内容的输出: + ```output + -Chicken Stew with Potatoes and Carrots: 3 tablespoons oil, 1 onion, chopped, 2 cloves garlic, minced, 1 carrot, peeled and chopped, 1 potato, peeled and chopped, 1 bay leaf, 1 thyme sprig, 1/2 teaspoon salt, 1/4 teaspoon black pepper, 1 1/2 cups chicken broth, 1/2 cup dry white wine, 2 tablespoons chopped fresh parsley, 2 tablespoons unsalted butter, 1 1/2 pounds boneless, skinless chicken thighs, cut into 1-inch pieces + -Oven-Roasted Chicken with Potatoes and Carrots: 3 tablespoons extra-virgin olive oil, 1 tablespoon Dijon mustard, 1 tablespoon chopped fresh rosemary, 1 tablespoon chopped fresh thyme, 4 cloves garlic, minced, 1 1/2 pounds small red potatoes, quartered, 1 1/2 pounds carrots, quartered lengthwise, 1/2 teaspoon salt, 1/4 teaspoon black pepper, 1 (4-pound) whole chicken + -Chicken, Potato, and Carrot Casserole: cooking spray, 1 large onion, chopped, 2 cloves garlic, minced, 1 carrot, peeled and shredded, 1 potato, peeled and shredded, 1/2 teaspoon dried thyme leaves, 1/4 teaspoon salt, 1/4 teaspoon black pepper, 2 cups fat-free, low-sodium chicken broth, 1 cup frozen peas, 1/4 cup all-purpose flour, 1 cup 2% reduced-fat milk, 1/4 cup grated Parmesan cheese - ```output - -Chicken Stew with Potatoes and Carrots: 3 tablespoons oil, 1 onion, chopped, 2 cloves garlic, minced, 1 carrot, peeled and chopped, 1 potato, peeled and chopped, 1 bay leaf, 1 thyme sprig, 1/2 teaspoon salt, 1/4 teaspoon black pepper, 1 1/2 cups chicken broth, 1/2 cup dry white wine, 2 tablespoons chopped fresh parsley, 2 tablespoons unsalted butter, 1 1/2 pounds boneless, skinless chicken thighs, cut into 1-inch pieces - -Oven-Roasted Chicken with Potatoes and Carrots: 3 tablespoons extra-virgin olive oil, 1 tablespoon Dijon mustard, 1 tablespoon chopped fresh rosemary, 1 tablespoon chopped fresh thyme, 4 cloves garlic, minced, 1 1/2 pounds small red potatoes, quartered, 1 1/2 pounds carrots, quartered lengthwise, 1/2 teaspoon salt, 1/4 teaspoon black pepper, 1 (4-pound) whole chicken - -Chicken, Potato, and Carrot Casserole: cooking spray, 1 large onion, chopped, 2 cloves garlic, minced, 1 carrot, peeled and shredded, 1 potato, peeled and shredded, 1/2 teaspoon dried thyme leaves, 1/4 teaspoon salt, 1/4 teaspoon black pepper, 2 cups fat-free, low-sodium chicken broth, 1 cup frozen peas, 1/4 cup all-purpose flour, 1 cup 2% reduced-fat milk, 1/4 cup grated Parmesan cheese - - -One Pot Chicken and Potato Dinner: 2 tablespoons olive oil, 1 pound boneless, skinless chicken thighs, cut into 1-inch pieces, 1 large onion, chopped, 3 cloves garlic, minced, 1 carrot, peeled and chopped, 1 potato, peeled and chopped, 1 bay leaf, 1 thyme sprig, 1/2 teaspoon salt, 1/4 teaspoon black pepper, 2 cups chicken broth, 1/2 cup dry white wine - - -Chicken, Potato, and Carrot Curry: 1 tablespoon vegetable oil, 1 large onion, chopped, 2 cloves garlic, minced, 1 carrot, peeled and chopped, 1 potato, peeled and chopped, 1 teaspoon ground coriander, 1 teaspoon ground cumin, 1/2 teaspoon ground turmeric, 1/2 teaspoon ground ginger, 1/4 teaspoon cayenne pepper, 2 cups chicken broth, 1/2 cup dry white wine, 1 (15-ounce) can chickpeas, drained and rinsed, 1/2 cup raisins, 1/2 cup chopped fresh cilantro - ``` + -One Pot Chicken and Potato Dinner: 2 tablespoons olive oil, 1 pound boneless, skinless chicken thighs, cut into 1-inch pieces, 1 large onion, chopped, 3 cloves garlic, minced, 1 carrot, peeled and chopped, 1 potato, peeled and chopped, 1 bay leaf, 1 thyme sprig, 1/2 teaspoon salt, 1/4 teaspoon black pepper, 2 cups chicken broth, 1/2 cup dry white wine + + -Chicken, Potato, and Carrot Curry: 1 tablespoon vegetable oil, 1 large onion, chopped, 2 cloves garlic, minced, 1 carrot, peeled and chopped, 1 potato, peeled and chopped, 1 teaspoon ground coriander, 1 teaspoon ground cumin, 1/2 teaspoon ground turmeric, 1/2 teaspoon ground ginger, 1/4 teaspoon cayenne pepper, 2 cups chicken broth, 1/2 cup dry white wine, 1 (15-ounce) can chickpeas, drained and rinsed, 1/2 cup raisins, 1/2 cup chopped fresh cilantro + ``` - > NOTE, your LLM is nondeterministic, so you might get different results every time you run the program. + > NOTE, your LLM is nondeterministic, so you might get different results every time you run the program. - Great, let's see how we can improve things. To improve things, we want to make sure the code is flexible, so ingredients and number of recipes can be improved and changed. + Great, let's see how we can improve things. To improve things, we want to make sure the code is flexible, so ingredients and number of recipes can be improved and changed. 3. 让我们按以下方式更改代码: - ```python - no_recipes = input("No of recipes (for example, 5: ") + ```python + no_recipes = input("No of recipes (for example, 5: ") - ingredients = input("List of ingredients (for example, chicken, potatoes, and carrots: ") - - # interpolate the number of recipes into the prompt an ingredients - prompt = f"Show me {no_recipes} recipes for a dish with the following ingredients: {ingredients}. Per recipe, list all the ingredients used" - ``` + ingredients = input("List of ingredients (for example, chicken, potatoes, and carrots: ") - Taking the code for a test run, could look like this: + # interpolate the number of recipes into the prompt an ingredients + prompt = f"Show me {no_recipes} recipes for a dish with the following ingredients: {ingredients}. Per recipe, list all the ingredients used" + ``` - ```output - No of recipes (for example, 5: 3 - List of ingredients (for example, chicken, potatoes, and carrots: milk,strawberries + Taking the code for a test run, could look like this: - -Strawberry milk shake: milk, strawberries, sugar, vanilla extract, ice cubes - -Strawberry shortcake: milk, flour, baking powder, sugar, salt, unsalted butter, strawberries, whipped cream - -Strawberry milk: milk, strawberries, sugar, vanilla extract - ``` + ```output + No of recipes (for example, 5: 3 + List of ingredients (for example, chicken, potatoes, and carrots: milk,strawberries -### 通过添加过滤器和购物清单进行改进 + -Strawberry milk shake: milk, strawberries, sugar, vanilla extract, ice cubes + -Strawberry shortcake: milk, flour, baking powder, sugar, salt, unsalted butter, strawberries, whipped cream + -Strawberry milk: milk, strawberries, sugar, vanilla extract + ``` +### 通过添加过滤器和购物清单进行改进 我们现在有一个能够生成食谱的应用,并且它使用很方便,基于用户的输入和食谱的数量所使用的成分。 @@ -437,171 +432,173 @@ That's your five recipes, with no garlic mentioned and you also have a shopping - **过滤掉成分**。 我们希望能够过滤掉我们不喜欢或过敏的成分。 要完成此更改,我们可以编辑现有提示并在其末尾添加过滤条件,如下所示: - ```python - filter = input("Filter (for example, vegetarian, vegan, or gluten-free: ") + ```python + filter = input("Filter (for example, vegetarian, vegan, or gluten-free: ") - prompt = f"Show me {no_recipes} recipes for a dish with the following ingredients: {ingredients}. Per recipe, list all the ingredients used, no {filter}" - ``` + prompt = f"Show me {no_recipes} recipes for a dish with the following ingredients: {ingredients}. Per recipe, list all the ingredients used, no {filter}" + ``` - Above, we add `{filter}` to the end of the prompt and we also capture the filter value from the user. - - An example input of running the program can now look like so: - - ```output - No of recipes (for example, 5: 3 - List of ingredients (for example, chicken, potatoes, and carrots: onion,milk - Filter (for example, vegetarian, vegan, or gluten-free: no milk - - 1. French Onion Soup - - Ingredients: - - -1 large onion, sliced - -3 cups beef broth - -1 cup milk - -6 slices french bread - -1/4 cup shredded Parmesan cheese - -1 tablespoon butter - -1 teaspoon dried thyme - -1/4 teaspoon salt - -1/4 teaspoon black pepper - - Instructions: - - 2. In a large pot, sauté onions in butter until golden brown. - 3. Add beef broth, milk, thyme, salt, and pepper. Bring to a boil. - 4. Reduce heat and simmer for 10 minutes. - 5. Place french bread slices on soup bowls. - 6. Ladle soup over bread. - 7. Sprinkle with Parmesan cheese. - - 8. Onion and Potato Soup - - Ingredients: - - -1 large onion, chopped - -2 cups potatoes, diced - -3 cups vegetable broth - -1 cup milk - -1/4 teaspoon black pepper - - Instructions: - - 9. In a large pot, sauté onions in butter until golden brown. - 10. Add potatoes, vegetable broth, milk, and pepper. Bring to a boil. - 11. Reduce heat and simmer for 10 minutes. - 12. Serve hot. - - 13. Creamy Onion Soup - - Ingredients: - - -1 large onion, chopped - -3 cups vegetable broth - -1 cup milk - -1/4 teaspoon black pepper - -1/4 cup all-purpose flour - -1/2 cup shredded Parmesan cheese - - Instructions: - - 14. In a large pot, sauté onions in butter until golden brown. - 15. Add vegetable broth, milk, and pepper. Bring to a boil. - 16. Reduce heat and simmer for 10 minutes. - 17. In a small bowl, whisk together flour and Parmesan cheese until smooth. - 18. Add to soup and simmer for an additional 5 minutes, or until soup has thickened. - ``` + Above, we add `{filter}` to the end of the prompt and we also capture the filter value from the user. - 正如您所看到的,任何含有牛奶的食谱都已被过滤掉。 但是,如果您患有乳糖不耐症,您可能也想过滤掉含有奶酪的食谱,因此有必要明确一下。 + An example input of running the program can now look like so: - ```python + ```output + No of recipes (for example, 5: 3 + List of ingredients (for example, chicken, potatoes, and carrots: onion,milk + Filter (for example, vegetarian, vegan, or gluten-free: no milk + + 1. French Onion Soup + + Ingredients: + + -1 large onion, sliced + -3 cups beef broth + -1 cup milk + -6 slices french bread + -1/4 cup shredded Parmesan cheese + -1 tablespoon butter + -1 teaspoon dried thyme + -1/4 teaspoon salt + -1/4 teaspoon black pepper + + Instructions: + + 2. In a large pot, sauté onions in butter until golden brown. + 3. Add beef broth, milk, thyme, salt, and pepper. Bring to a boil. + 4. Reduce heat and simmer for 10 minutes. + 5. Place french bread slices on soup bowls. + 6. Ladle soup over bread. + 7. Sprinkle with Parmesan cheese. + + 8. Onion and Potato Soup + + Ingredients: + + -1 large onion, chopped + -2 cups potatoes, diced + -3 cups vegetable broth + -1 cup milk + -1/4 teaspoon black pepper + + Instructions: + + 9. In a large pot, sauté onions in butter until golden brown. + 10. Add potatoes, vegetable broth, milk, and pepper. Bring to a boil. + 11. Reduce heat and simmer for 10 minutes. + 12. Serve hot. + + 13. Creamy Onion Soup + + Ingredients: + + -1 large onion, chopped + -3 cups vegetable broth + -1 cup milk + -1/4 teaspoon black pepper + -1/4 cup all-purpose flour + -1/2 cup shredded Parmesan cheese + + Instructions: + + 14. In a large pot, sauté onions in butter until golden brown. + 15. Add vegetable broth, milk, and pepper. Bring to a boil. + 16. Reduce heat and simmer for 10 minutes. + 17. In a small bowl, whisk together flour and Parmesan cheese until smooth. + 18. Add to soup and simmer for an additional 5 minutes, or until soup has thickened. + ``` + + 正如您所看到的,任何含有牛奶的食谱都已被过滤掉。 但是,如果您患有乳糖不耐症,您可能也想过滤掉含有奶酪的食谱,因此有必要明确一下。 + + ```python + + ``` - **制作购物清单**。 我们想根据家里已有的物品制定一份购物清单。 - 对于此功能,我们可以尝试在一个提示中解决所有问题,也可以将其分成两个提示。 让我们尝试一下后一种方法。 在这里,我们建议添加一个额外的提示,但为了使其起作用,我们需要将前一个提示的结果作为上下文添加到后一个提示中。 + 对于此功能,我们可以尝试在一个提示中解决所有问题,也可以将其分成两个提示。 让我们尝试一下后一种方法。 在这里,我们建议添加一个额外的提示,但为了使其起作用,我们需要将前一个提示的结果作为上下文添加到后一个提示中。 - 找到代码中打印第一个提示结果的部分,并添加以下代码: + 找到代码中打印第一个提示结果的部分,并添加以下代码: - ```python - old_prompt_result = completion.choices[0].text - prompt = "Produce a shopping list for the generated recipes and please don't include ingredients that I already have." - - new_prompt = f"{old_prompt_result} {prompt}" - completion = openai.Completion.create(engine=deployment_name, prompt=new_prompt, max_tokens=1200) - - # print response - print("Shopping list:") - print(completion.choices[0].text) - ``` + ```python + old_prompt_result = completion.choices[0].text + prompt = "Produce a shopping list for the generated recipes and please don't include ingredients that I already have." + + new_prompt = f"{old_prompt_result} {prompt}" + completion = openai.Completion.create(engine=deployment_name, prompt=new_prompt, max_tokens=1200) - 请注意以下事项: + # print response + print("Shopping list:") + print(completion.choices[0].text) + ``` - 1. 我们通过将第一个提示的结果添加到新提示来构造一个新提示 + 请注意以下事项: - ```python - new_prompt = f"{old_prompt_result} {prompt}" - ``` + 1. 我们通过将第一个提示的结果添加到新提示来构造一个新提示 - 2. 我们提出一个新的请求,但也考虑到我们在第一个提示中请求的 token 数量,所以这次我们说 `max_tokens` 是1200。 + ```python + new_prompt = f"{old_prompt_result} {prompt}" + ``` - ```python - completion = openai.Completion.create(engine=deployment_name, prompt=new_prompt, max_tokens=1200) - ``` + 2. 我们提出一个新的请求,但也考虑到我们在第一个提示中请求的 token 数量,所以这次我们说 `max_tokens` 是 1200。 - 运行代码,结果如下: + ```python + completion = openai.Completion.create(engine=deployment_name, prompt=new_prompt, max_tokens=1200) + ``` - ```output - No of recipes (for example, 5: 2 - List of ingredients (for example, chicken, potatoes, and carrots: apple,flour - Filter (for example, vegetarian, vegan, or gluten-free: sugar - Recipes: - or milk. - - -Apple and flour pancakes: 1 cup flour, 1/2 tsp baking powder, 1/2 tsp baking soda, 1/4 tsp salt, 1 tbsp sugar, 1 egg, 1 cup buttermilk or sour milk, 1/4 cup melted butter, 1 Granny Smith apple, peeled and grated - -Apple fritters: 1-1/2 cups flour, 1 tsp baking powder, 1/4 tsp salt, 1/4 tsp baking soda, 1/4 tsp nutmeg, 1/4 tsp cinnamon, 1/4 tsp allspice, 1/4 cup sugar, 1/4 cup vegetable shortening, 1/4 cup milk, 1 egg, 2 cups shredded, peeled apples - Shopping list: - -Flour, baking powder, baking soda, salt, sugar, egg, buttermilk, butter, apple, nutmeg, cinnamon, allspice - ``` + 运行代码,结果如下: + + ```output + No of recipes (for example, 5: 2 + List of ingredients (for example, chicken, potatoes, and carrots: apple,flour + Filter (for example, vegetarian, vegan, or gluten-free: sugar + Recipes: + or milk. + + -Apple and flour pancakes: 1 cup flour, 1/2 tsp baking powder, 1/2 tsp baking soda, 1/4 tsp salt, 1 tbsp sugar, 1 egg, 1 cup buttermilk or sour milk, 1/4 cup melted butter, 1 Granny Smith apple, peeled and grated + -Apple fritters: 1-1/2 cups flour, 1 tsp baking powder, 1/4 tsp salt, 1/4 tsp baking soda, 1/4 tsp nutmeg, 1/4 tsp cinnamon, 1/4 tsp allspice, 1/4 cup sugar, 1/4 cup vegetable shortening, 1/4 cup milk, 1 egg, 2 cups shredded, peeled apples + Shopping list: + -Flour, baking powder, baking soda, salt, sugar, egg, buttermilk, butter, apple, nutmeg, cinnamon, allspice + ``` ## 改进您的设置 到目前为止,我们拥有的是可以运行的代码,但是我们应该做一些调整来进一步改进。 我们应该做的一些事情是: -- **将 Key 与代码分开**,例如 API Key。Key 不属于代码,应存储在安全的位置。 为了将 Key 与代码分开,我们可以使用环境变量和像`python-dotenv` 这样的 libraries 从文件中加载它们。 代码如下: +- **将 Key 与代码分开**,例如 API Key。Key 不属于代码,应存储在安全的位置。 为了将 Key 与代码分开,我们可以使用环境变量和像`python-dotenv` 这样的 libraries 从文件中加载它们。 代码如下: - 1. 创建一个包含以下内容的 `.env` 文件: + 1. 创建一个包含以下内容的 `.env` 文件: - ```bash - OPENAI_API_KEY=sk-... - ``` + ```bash + OPENAI_API_KEY=sk-... + ``` - > Note, for Azure, you need to set the following environment variables: + > Note, for Azure, you need to set the following environment variables: - ```bash - OPENAI_API_TYPE=azure - OPENAI_API_VERSION=2023-05-15 - OPENAI_API_BASE= - ``` + ```bash + OPENAI_API_TYPE=azure + OPENAI_API_VERSION=2023-05-15 + OPENAI_API_BASE= + ``` - In code, you would load the environment variables like so: + In code, you would load the environment variables like so: - ```python - from dotenv import load_dotenv + ```python + from dotenv import load_dotenv - load_dotenv() + load_dotenv() - openai.api_key = os.environ["OPENAI_API_KEY"] - ``` + openai.api_key = os.environ["OPENAI_API_KEY"] + ``` -- **关于 token 长度**。 我们应该考虑需要多少 token 来生成我们想要的文本。 token 需要花钱,因此在可能的情况下,我们应该尽量节约使用 token 的数量。 例如,我们可以对提示进行调整,以便我们可以使用更少的 token +- **关于 token 长度**。 我们应该考虑需要多少 token 来生成我们想要的文本。 token 需要花钱,因此在可能的情况下,我们应该尽量节约使用 token 的数量。 例如,我们可以对提示进行调整,以便我们可以使用更少的 token - 要更改使用的 token,您可以使用 `max_tokens` 参数。 例如,如果您想使用 100 个 token,您可以这样做: + 要更改使用的 token,您可以使用 `max_tokens` 参数。 例如,如果您想使用 100 个 token,您可以这样做: - ```python - completion = openai.Completion.create(model="davinci-002", prompt=prompt, max_tokens=100) - ``` + ```python + completion = openai.Completion.create(model="davinci-002", prompt=prompt, max_tokens=100) + ``` -**进行 temperature 调整试验**。 temperature 是我们到目前为止还没有提到的东西,但它是我们的程序如何执行的重要元素。 temperature 值越高,输出就越随机。 相反, temperature 值越低,输出就越可预测。 考虑一下您是否希望输出有所变化。 +**进行 temperature 调整试验**。 temperature 是我们到目前为止还没有提到的东西,但它是我们的程序如何执行的重要元素。 temperature 值越高,输出就越随机。 相反, temperature 值越低,输出就越可预测。 考虑一下您是否希望输出有所变化。 要改变 temperature ,您可以使用 `temperature` 参数。 例如,如果您想使用 0.5 的 temperature ,您可以这样做: @@ -609,7 +606,7 @@ That's your five recipes, with no garlic mentioned and you also have a shopping completion = openai.Completion.create(model="davinci-002", prompt=prompt, temperature=0.5) ``` - > 请注意,越接近 1.0,输出的变化就越多。 +> 请注意,越接近 1.0,输出的变化就越多。 ## 作业 @@ -667,5 +664,4 @@ A: 1 想要了解有关创建文本生成应用的更多信息? 转至[进阶学习的页面](../../../13-continued-learning/translations/cn/README.md?WT.mc_id=academic-105485-koreyst) 查找有关此主章节的其他学习资源。 -前往第七章,我们将学习[构建聊天应用程序](../../../07-building-chat-applications/translations/cn/README.md?WT.mc_id=academic-105485-koreyst) - +前往第七章,我们将学习[构建聊天应用程序](../../../07-building-chat-applications/translations/cn/README.md?WT.mc_id=academic-105485-koreyst) diff --git a/06-text-generation-apps/translations/ja-jp/README.md b/06-text-generation-apps/translations/ja-jp/README.md index 3626be1e3..4b0a834a5 100644 --- a/06-text-generation-apps/translations/ja-jp/README.md +++ b/06-text-generation-apps/translations/ja-jp/README.md @@ -1,10 +1,10 @@ # テキスト生成アプリケーションの構築 -[![Building Text Generation Applications](../../images/06-lesson-banner.png?WT.mc_id=academic-105485-yoterada)](https://youtu.be/5jKHzY6-4s8?WT.mc_id=academic-105485-yoterada) +[![Building Text Generation Applications](../../images/06-lesson-banner.png?WT.mc_id=academic-105485-yoterada)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=bf3f3528-9871-4628-8616-b4b03cb23dcdt?WT.mc_id=academic-105485-yoterada) -> *(上記の画像をクリックすると、レッスン・ビデオを表示します)* +> _(上記の画像をクリックすると、レッスン・ビデオを表示します)_ -これまでのレッスンを通じて、プロンプトの基本的な概念や、「プロンプト・エンジニアリング」と呼ぶテクニックについて学んできました。ChatGPT、Office 365、Microsoft Power Platform など、プロンプトで操作可能な多くのツールは、皆様が何かを成し遂げるためのサポートを行います。 +これまでのレッスンを通じて、プロンプトの基本的な概念や、「プロンプト・エンジニアリング」と呼ぶテクニックについて学んできました。ChatGPT、Office 365、Microsoft Power Platform  など、プロンプトで操作可能な多くのツールは、皆様が何かを成し遂げるためのサポートを行います。 皆様のアプリケーションに、このような機能を追加するためには、プロンプトや Completion といった概念を理解し、使用するライブラリを選択する必要があります。この章では、それらを具体的に学んでいきます。 @@ -54,21 +54,21 @@ ## どこから始めればよいのでしょうか? -まず、LLM と統合する方法を理解する必要があります。これには通常、以下の2つのアプローチが含まれます: +まず、LLM と統合する方法を理解する必要があります。これには通常、以下の 2 つのアプローチが含まれます: - API の使用:プロンプトを含む HTTP (RESTful) リクエストを作成し、生成されたテキストを取得 - ライブラリの使用:上記の API 呼び出しより簡単に利用可能で、API 呼び出しを隠蔽したライブラリを使用 ## ライブラリ/SDK -LLMを操作するために、いくつかの有名なライブラリがあります: +LLM を操作するために、いくつかの有名なライブラリがあります: - **OpenAI** このライブラリを使用すると、モデルに対する接続とプロンプトの送信が簡単になります また、より高レベルで動作するライブラリもあります: - **Langchain**:Langchain はよく知られており、Python をサポートしています -- **Semantic Kernel**:Semantic Kernelは、C#、Python、Java をサポートするMicrosoft 純正のライブラリです +- **Semantic Kernel**:Semantic Kernel は、C#、Python、Java をサポートする Microsoft 純正のライブラリです ## OpenAI を使用した最初のアプリ @@ -84,16 +84,16 @@ pip install openai ### リソースの作成 -下記の手順に従い操作を行なってください: +下記の手順に従い操作を行なってください: -- Azureのアカウントを作成します。[https://azure.microsoft.com/free/](https://azure.microsoft.com/free/?WT.mc_id=academic-105485-yoterada) +- Azure のアカウントを作成します。[https://azure.microsoft.com/free/](https://azure.microsoft.com/free/?WT.mc_id=academic-105485-yoterada) - Azure Open AI へのアクセス権限を取得します。[https://learn.microsoft.com/azure/ai-services/openai/overview#how-do-i-get-access-to-azure-openai](https://learn.microsoft.com/azure/ai-services/openai/overview#how-do-i-get-access-to-azure-openai?WT.mc_id=academic-105485-yoterada) からアクセス申請を提出します - > [!NOTE] - > 本記事の執筆時には、Azure Open AI へのアクセスを申請する必要があります。 + > [!NOTE] + > 本記事の執筆時には、Azure Open AI へのアクセスを申請する必要があります。 -- Python をインストールします。 -- Azure OpenAI Serviceリソースを作成します。[リソースの作成方法](https://learn.microsoft.com/azure/ai-services/openai/how-to/create-resource?pivots=web-portal?WT.mc_id=academic-105485-yoterada)については、こちらのガイドをご参照ください。 +- Python をインストールします。 +- Azure OpenAI Service リソースを作成します。[リソースの作成方法](https://learn.microsoft.com/azure/ai-services/openai/how-to/create-resource?pivots=web-portal?WT.mc_id=academic-105485-yoterada)については、こちらのガイドをご参照ください。 ### API キーとエンドポイントを取得 @@ -107,7 +107,7 @@ pip install openai > API キーはコード中に直接書き込まずに管理すべきです。下記のように環境変数を使用して設定できます。 > > - 環境変数 `OPENAI_API_KEY` に API キーを設定します -> `export OPENAI_API_KEY='sk-...'` +> `export OPENAI_API_KEY='sk-...'` ### Azure の設定 @@ -120,15 +120,14 @@ openai.api_version = '2023-05-15' openai.api_base = os.getenv("API_BASE") ``` -上記は、それぞれ下記の設定を行っています: +上記は、それぞれ下記の設定を行っています: -- `api_type`を`azure`に設定します。これにより、ライブラリは OpenAI ではなく Azure Open AI に接続します。 -- `api_key`は、Azure Portal で取得した API キーを設定します。 -- `api_version`は、使用する API のバージョンです。執筆時点での最新バージョンは`2023-05-15`です。 +- `api_type`を`azure`に設定します。これにより、ライブラリは OpenAI ではなく Azure Open AI に接続します。 +- `api_key`は、Azure Portal で取得した API キーを設定します。 +- `api_version`は、使用する API のバージョンです。執筆時点での最新バージョンは`2023-05-15`です。 - `api_base` は API のエンドポイントです。これは、Azure Portal の API キーの下側に記載されています。 -> [!NOTE] -> `os.getenv` は環境変数を読み取る関数です。`OPENAI_API_KEY` や `API_BASE` などの環境変数を読み取るために使用します。これらの環境変数は、ターミナルで設定するか、もしくは `dotenv` のようなライブラリを使用して設定します。 +> [!NOTE] > `os.getenv` は環境変数を読み取る関数です。`OPENAI_API_KEY` や `API_BASE` などの環境変数を読み取るために使用します。これらの環境変数は、ターミナルで設定するか、もしくは `dotenv` のようなライブラリを使用して設定します。 > [!NOTE] > 訳者追記 @@ -183,14 +182,14 @@ pip install openai > [!NOTE] > Azure OpenAI キーは、[https://portal.azure.com/](https://portal.azure.com/?WT.mc_id=academic-105485-yoterada) から `OpenAI` を検索し、`OpenAI Resource` を選択してください。その後 `Keys and Endpoint` を選択し、`Key 1` の値をコピーてください。 -1. *app.py* ファイルを作成し、下記のコードを記述してください: +1. _app.py_ ファイルを作成し、下記のコードを記述してください: ```python import openai openai.api_key = "" -openai.api_type = 'azure' +openai.api_type = 'azure' openai.api_version = '2023-05-15' openai.api_base = "" deployment_name = "<デプロイ名>" @@ -200,7 +199,7 @@ prompt = "次の文を完成させてください: むかしむかし、ある # Completion を作成 completion = openai.Completion.create(engine= deployment_name, model="davinci-002", prompt=prompt) - + # レスポンスを出力 print(completion.choices[0].text) ``` @@ -237,7 +236,7 @@ print(completion.choices[0].text) 上記のプロンプトを使用すると、以下のようなレスポンスが得られるかもしれません: ```output -1. ローストチキンと野菜: +1. ローストチキンと野菜: 材料: - チキンのもも肉 4 枚 - じゃがいも 2 個、角切り @@ -380,7 +379,7 @@ print(completion.choices[0].text) シナリオを一通り試したところで、そのシナリオに合わせたコードを実装してみましょう。下記の手順に従い実装してください。 -1. 既存の *app.py* ファイルを利用します。 +1. 既存の _app.py_ ファイルを利用します。 2. `prompt`という変数を探し、下記のようにコードを修正してください。 ```python @@ -401,7 +400,7 @@ prompt = f"鶏肉、じゃがいも、にんじんを使った料理のレシピ -鶏肉、じゃがいも、人参のカレー: サラダ油大さじ1, 大きめの玉ねぎ1個(みじん切り), ニンニク2片(みじん切り), 人参1本(皮をむいてみじん切り), じゃがいも1個(皮をむいてみじん切り), コリアンダー(パウダー)小さじ1, クミン(パウダー)小さじ1, ターメリック(パウダー)小さじ1/2, ジンジャー(パウダー)小さじ1/2, ケイエンペッパー小さじ1/4, 鶏ガラスープ2カップ, ドライホワイトワイン1/2カップ, ヒヨコ豆1缶(15オンス、水切り), レーズン1/2カップ, シラントロ(みじん切り)1/2カップ ``` -> 注意: LLM は非決定的(非確実)なので、プログラムを実行するたびに上記とは異なる結果が得られるかもしれません。 +> 注意: LLM  は非決定的(非確実)なので、プログラムを実行するたびに上記とは異なる結果が得られるかもしれません。 さてそれでは、どうすれば改善できるのかを見てみましょう。改善するために、コードに柔軟性を持たせ、材料やレシピの数を変更できるようにします。 @@ -435,7 +434,7 @@ prompt = f"{ingredients} を使った料理のレシピを {no_recipes} つ教 - **材料のフィルタリング**。嫌いな食材やアレルギーのある食材をフィルタリングできるようにしたいと考えています。これを実現するために、既存のプロンプトを編集し、下記のようにフィルター条件をプロンプトの最後に追加してください。 -```python +```python filter = input("フィルター(例:ベジタリアン、ビーガン、グルテンフリー): ") prompt = f"{ingredients} を使った料理のレシピを {no_recipes} つ教えてください。レシピごとに、使用するすべての食材をリストしてください。ただし、{filter}は除外してください。" @@ -529,7 +528,7 @@ print(completion.choices[0].text) new_prompt = f"{old_prompt_result} {prompt}" ``` -1. 新しいリクエストを作成しますが、最初のプロンプトで要求したトークン数も考慮に入れて、今回は `max_tokens` を1200とします。 +1. 新しいリクエストを作成しますが、最初のプロンプトで要求したトークン数も考慮に入れて、今回は `max_tokens` を 1200 とします。 ```python completion = openai.Completion.create(engine=deployment_name, prompt=new_prompt, max_tokens=1200) @@ -537,7 +536,7 @@ completion = openai.Completion.create(engine=deployment_name, prompt=new_prompt, このコードを実行すると、下記のような結果が得られます: -```output +```output レシピの数(例えば、5): 2 食材のリスト(例えば、チキン、ポテト、キャロット): リンゴ、小麦粉 フィルター(例えば、ベジタリアン、ビーガン、グルテンフリー): 砂糖 @@ -565,7 +564,7 @@ completion = openai.Completion.create(engine=deployment_name, prompt=new_prompt, OPENAI_API_KEY=sk-... ``` -> ご注意:Azureを使用する場合は、下記の環境変数を設定する必要があります: +> ご注意:Azure を使用する場合は、下記の環境変数を設定する必要があります: ```bash OPENAI_API_TYPE=azure @@ -593,13 +592,13 @@ completion = openai.Completion.create(model="davinci-002", prompt=prompt, max_to - **温度 (Temperature)** これまで触れていなかった温度ですが、プログラムの実行結果に大きな影響を与えます。温度が高いほど(値が 1 に近いほど)、出力はランダムになります。逆に、温度値が低いほど(値が 0 に近いほど)、出力結果は予測可能になります。ご自身のアプリで出力にバリエーションが必要かどうかを考えてください。 -温度を変更するには、`temperature` パラメータを使用します。例えば、温度を0.5に設定したい場合は、以下のようにします: +温度を変更するには、`temperature` パラメータを使用します。例えば、温度を 0.5 に設定したい場合は、以下のようにします: ```python completion = openai.Completion.create(model="davinci-002", prompt=prompt, temperature=0.5) ``` -> 注意:1.0に近づくほど、出力のバリエーションが大きくなります。 +> 注意:1.0 に近づくほど、出力のバリエーションが大きくなります。 ## 課題 @@ -618,13 +617,13 @@ completion = openai.Completion.create(model="davinci-002", prompt=prompt, temper 下記は、あくまでも参考のプロンプトですので、使い方を確認し、お好みに合わせて微調整してください。 ```text -- "あなたは Python 言語のエキスパートです - -下記の形式で Python の初心者用レッスンを提案してください: - -形式: - - 概念: - - レッスンの簡単な説明: +- "あなたは Python 言語のエキスパートです + +下記の形式で Python の初心者用レッスンを提案してください: + +形式: + - 概念: + - レッスンの簡単な説明: - 解答付きのコード演習" ``` diff --git a/06-text-generation-apps/translations/pt-br/README.md b/06-text-generation-apps/translations/pt-br/README.md index 2049e3a9a..ece71c8bc 100644 --- a/06-text-generation-apps/translations/pt-br/README.md +++ b/06-text-generation-apps/translations/pt-br/README.md @@ -1,8 +1,8 @@ # Desenvolvendo Aplicativos de Geração de Texto -[![Building Text Generation Applications](../../images/06-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/5jKHzY6-4s8?WT.mc_id=academic-105485-koreyst) +[![Building Text Generation Applications](../../images/06-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=bf3f3528-9871-4628-8616-b4b03cb23dcd?WT.mc_id=academic-105485-koreyst) -> *(Clique na imagem acima para assistir ao vídeo desta lição)* +> _(Clique na imagem acima para assistir ao vídeo desta lição)_ Você viu até agora neste currículo que existem conceitos básicos como prompts e até mesmo uma disciplina inteira chamada "engenharia de prompts". Muitas ferramentas com as quais você pode interagir, como ChatGPT, Office 365, Microsoft Power Platform e muito mais, suportam o uso de prompts para realizar algo. @@ -107,7 +107,7 @@ Agora que você tem essas informações copiadas, vamos instruir as bibliotecas > Vale a pena separar sua chave API do seu código. Você pode fazer isso usando variáveis de ambiente. > > - Defina a variável de ambiente em: `OPENAI_API_TYPE` para a chave da sua API. -> `export OPENAI_API_KEY='sk-...'` +> `export OPENAI_API_KEY='sk-...'` ### Configurando o recurso do Azure no código @@ -129,8 +129,7 @@ Aqui estamos definindo o seguinte: - `api_version`, esta é a versão da API que você deseja usar. No momento da escrita, a versão mais recente é `2023-05-15`. - `api_base`, este é o endpoint da API. Você pode encontrá-lo no Portal do Azure ao lado de sua chave API. -> [!OBSERVAÇÃO] -> `os.getenv` é uma função que lê variáveis de ambiente. Você pode usá-lo para ler variáveis de ambiente como `OPENAI_API_KEY` e `API_BASE`. Defina essas variáveis de ambiente em seu terminal ou usando uma biblioteca como `dotenv`. +> [!OBSERVAÇÃO] > `os.getenv` é uma função que lê variáveis de ambiente. Você pode usá-lo para ler variáveis de ambiente como `OPENAI_API_KEY` e `API_BASE`. Defina essas variáveis de ambiente em seu terminal ou usando uma biblioteca como `dotenv`. ## Gerando texto @@ -166,50 +165,50 @@ Agora que aprendemos a configurar o openai, é hora de criar o seu primeiro apli 1. Crie um ambiente virtual e instale o openai: - ```bash - python -m venv venv - source venv/bin/activate - pip install openai - ``` + ```bash + python -m venv venv + source venv/bin/activate + pip install openai + ``` + + > [!OBSERVAÇÃO] + > Se você estiver usando o Windows, digite `venv\Scripts\activate` em vez de `source venv/bin/activate`. - > [!OBSERVAÇÃO] - > Se você estiver usando o Windows, digite `venv\Scripts\activate` em vez de `source venv/bin/activate`. + > [!OBSERVAÇÃO] + > Localize sua chave Azure Open AI acessando [https://portal.azure.com/](https://portal.azure.com/?WT.mc_id=academic-105485-koreyst) e pesquise por `Open AI` e selecione o `recurso Open AI` e depois selecione `Chaves e Endpoint` e copie o valor `Chave 1`. - > [!OBSERVAÇÃO] - > Localize sua chave Azure Open AI acessando [https://portal.azure.com/](https://portal.azure.com/?WT.mc_id=academic-105485-koreyst) e pesquise por `Open AI` e selecione o `recurso Open AI` e depois selecione `Chaves e Endpoint` e copie o valor `Chave 1`. +1. Crie um arquivo _app.py_ e dê a ele o seguinte código: -1. Crie um arquivo *app.py* e dê a ele o seguinte código: + ```python + import openai - ```python - import openai + openai.api_key = "" - openai.api_key = "" + openai.api_type = 'azure' + openai.api_version = '2023-05-15' + openai.api_base = "" + deployment_name = "" - openai.api_type = 'azure' - openai.api_version = '2023-05-15' - openai.api_base = "" - deployment_name = "" + # add your completion code + prompt = "Complete the following: Once upon a time there was a" - # add your completion code - prompt = "Complete the following: Once upon a time there was a" + # make completion + completion = openai.Completion.create(engine= deployment_name, model="davinci-002", prompt=prompt) - # make completion - completion = openai.Completion.create(engine= deployment_name, model="davinci-002", prompt=prompt) - - # print response - print(completion.choices[0].text) - ``` + # print response + print(completion.choices[0].text) + ``` - > [!OBSERVAÇÃO] - > Se você estiver usando o Azure Open AI, precisará definir o `api_type` como `azure` e definir o `api_key` como sua chave Azure Open AI. + > [!OBSERVAÇÃO] + > Se você estiver usando o Azure Open AI, precisará definir o `api_type` como `azure` e definir o `api_key` como sua chave Azure Open AI. - Você deve ver uma saída como a seguinte: + Você deve ver uma saída como a seguinte: - ```output - very unhappy _____. + ```output + very unhappy _____. - Once upon a time there was a very unhappy mermaid. - ``` + Once upon a time there was a very unhappy mermaid. + ``` ## Diferentes tipos de prompts, para diferentes coisas @@ -351,7 +350,7 @@ Agora você tem um novo resultado, a saber: - 1 colher de chá de tomilho seco - 4 xícaras de caldo de galinha - Sal e pimenta a gosto - + 5. Hash de Frango e Batata: - Ingredientes: - 2 colheres de sopa de azeite de oliva @@ -376,54 +375,53 @@ Aqui estão as cinco receitas, sem mencionar alho, e também uma lista de compra Agora que temos um cenário, vamos escrever o código para corresponder ao cenário demonstrado. Para fazer isso, siga estas etapas: - -1. Use o arquivo *app.py* existente como ponto de partida +1. Use o arquivo _app.py_ existente como ponto de partida 1. Localize a variável `prompt` e altere seu código para o seguinte: - ```python - prompt = "Me forneça 5 receitas de um prato com os seguintes ingredientes: frango, batata e cenoura. Por receita, liste todos os ingredientes usados" - ``` + ```python + prompt = "Me forneça 5 receitas de um prato com os seguintes ingredientes: frango, batata e cenoura. Por receita, liste todos os ingredientes usados" + ``` - Se você executar o código agora, deverá ver uma saída semelhante a: + Se você executar o código agora, deverá ver uma saída semelhante a: - ```output - - Ensopado de Frango com Batata e Cenoura: 3 colheres de sopa de azeite, 1 cebola picada, 2 dentes de alho picados, 1 cenoura descascada e picada, 1 batata descascada e picada, 1 folha de louro, 1 raminho de tomilho, 1/2 colher de chá de sal , 1/4 colher de chá de pimenta preta, 1 1/2 xícara de caldo de galinha, 1/2 xícara de vinho branco seco, 2 colheres de sopa de salsa fresca picada, 2 colheres de sopa de manteiga sem sal, 1 1/2 libra de coxas de frango desossadas e sem pele, cortadas em 1- pedaços de polegada + ```output + - Ensopado de Frango com Batata e Cenoura: 3 colheres de sopa de azeite, 1 cebola picada, 2 dentes de alho picados, 1 cenoura descascada e picada, 1 batata descascada e picada, 1 folha de louro, 1 raminho de tomilho, 1/2 colher de chá de sal , 1/4 colher de chá de pimenta preta, 1 1/2 xícara de caldo de galinha, 1/2 xícara de vinho branco seco, 2 colheres de sopa de salsa fresca picada, 2 colheres de sopa de manteiga sem sal, 1 1/2 libra de coxas de frango desossadas e sem pele, cortadas em 1- pedaços de polegada - - Frango Assado no Forno com Batata e Cenoura: 3 colheres de sopa de azeite extra-virgem, 1 colher de sopa de mostarda Dijon, 1 colher de sopa de alecrim fresco picado, 1 colher de sopa de tomilho fresco picado, 4 dentes de alho picados, 1 1/2 libra de batatas vermelhas pequenas, esquartejado, 1 1/2 libra de cenoura, cortada em quartos longitudinalmente, 1/2 colher de chá de sal, 1/4 colher de chá de pimenta preta, 1 (4 libras) de frango inteiro + - Frango Assado no Forno com Batata e Cenoura: 3 colheres de sopa de azeite extra-virgem, 1 colher de sopa de mostarda Dijon, 1 colher de sopa de alecrim fresco picado, 1 colher de sopa de tomilho fresco picado, 4 dentes de alho picados, 1 1/2 libra de batatas vermelhas pequenas, esquartejado, 1 1/2 libra de cenoura, cortada em quartos longitudinalmente, 1/2 colher de chá de sal, 1/4 colher de chá de pimenta preta, 1 (4 libras) de frango inteiro - - Caçarola de frango, batata e cenoura: spray de cozinha, 1 cebola grande picada, 2 dentes de alho picados, 1 cenoura descascada e ralada, 1 batata descascada e ralada, 1/2 colher de chá de folhas secas de tomilho, 1/4 colher de chá sal, 1/4 colher de chá de pimenta preta, 2 xícaras de caldo de galinha desnatado e com baixo teor de sódio, 1 xícara de ervilhas congeladas, 1/4 xícara de farinha de trigo, 1 xícara de leite com baixo teor de gordura a 2%, 1/4 xícara de parmesão ralado queijo - - - Jantar de frango e batata em uma panela: 2 colheres de sopa de azeite, 1 quilo de coxas de frango desossadas e sem pele, cortadas em pedaços de 2,5 cm, 1 cebola grande picada, 3 dentes de alho picados, 1 cenoura descascada e picada, 1 batata, descascado e picado, 1 folha de louro, 1 raminho de tomilho, 1/2 colher de chá de sal, 1/4 colher de chá de pimenta preta, 2 xícaras de caldo de galinha, 1/2 xícara de vinho branco seco - - - Caril de Frango, Batata e Cenoura: 1 colher de sopa de óleo vegetal, 1 cebola grande picada, 2 dentes de alho picados, 1 cenoura descascada e picada, 1 batata descascada e picada, 1 colher de chá de coentro moído, 1 colher de chá de cominho moído, 1/2 colher de chá de açafrão em pó, 1/2 colher de chá de gengibre em pó, 1/4 colher de chá de pimenta caiena, 2 xícaras de caldo de galinha, 1/2 xícara de vinho branco seco, 1 lata (15 onças) de grão de bico, escorrido e enxaguado, 1/2 xícara de passas, 1/2 xícara de coentro fresco picado - ``` + - Caçarola de frango, batata e cenoura: spray de cozinha, 1 cebola grande picada, 2 dentes de alho picados, 1 cenoura descascada e ralada, 1 batata descascada e ralada, 1/2 colher de chá de folhas secas de tomilho, 1/4 colher de chá sal, 1/4 colher de chá de pimenta preta, 2 xícaras de caldo de galinha desnatado e com baixo teor de sódio, 1 xícara de ervilhas congeladas, 1/4 xícara de farinha de trigo, 1 xícara de leite com baixo teor de gordura a 2%, 1/4 xícara de parmesão ralado queijo - > [!OBSERVAÇÃO] - > seu LLM é não determinístico, então você pode obter resultados diferentes sempre que executar o programa. + - Jantar de frango e batata em uma panela: 2 colheres de sopa de azeite, 1 quilo de coxas de frango desossadas e sem pele, cortadas em pedaços de 2,5 cm, 1 cebola grande picada, 3 dentes de alho picados, 1 cenoura descascada e picada, 1 batata, descascado e picado, 1 folha de louro, 1 raminho de tomilho, 1/2 colher de chá de sal, 1/4 colher de chá de pimenta preta, 2 xícaras de caldo de galinha, 1/2 xícara de vinho branco seco - Ótimo! Vamos ver como podemos melhorar as coisas. Para melhorar as coisas, queremos ter certeza de que o código é flexível, para que os ingredientes e o número de receitas possam ser melhorados e alterados. + - Caril de Frango, Batata e Cenoura: 1 colher de sopa de óleo vegetal, 1 cebola grande picada, 2 dentes de alho picados, 1 cenoura descascada e picada, 1 batata descascada e picada, 1 colher de chá de coentro moído, 1 colher de chá de cominho moído, 1/2 colher de chá de açafrão em pó, 1/2 colher de chá de gengibre em pó, 1/4 colher de chá de pimenta caiena, 2 xícaras de caldo de galinha, 1/2 xícara de vinho branco seco, 1 lata (15 onças) de grão de bico, escorrido e enxaguado, 1/2 xícara de passas, 1/2 xícara de coentro fresco picado + ``` + + > [!OBSERVAÇÃO] + > seu LLM é não determinístico, então você pode obter resultados diferentes sempre que executar o programa. + + Ótimo! Vamos ver como podemos melhorar as coisas. Para melhorar as coisas, queremos ter certeza de que o código é flexível, para que os ingredientes e o número de receitas possam ser melhorados e alterados. 1. Vamos alterar o código da seguinte maneira: - ```python - no_recipes = input("No of recipes (for example, 5: ") + ```python + no_recipes = input("No of recipes (for example, 5: ") - ingredients = input("List of ingredients (for example, chicken, potatoes, and carrots: ") - - # interpolate the number of recipes into the prompt an ingredients - prompt = f"Show me {no_recipes} recipes for a dish with the following ingredients: {ingredients}. Per recipe, list all the ingredients used" - ``` + ingredients = input("List of ingredients (for example, chicken, potatoes, and carrots: ") - Execute o código e teste, e vejamos o resultado: + # interpolate the number of recipes into the prompt an ingredients + prompt = f"Show me {no_recipes} recipes for a dish with the following ingredients: {ingredients}. Per recipe, list all the ingredients used" + ``` - ```output - No of recipes (for example, 5: 3 - List of ingredients (for example, chicken, potatoes, and carrots: milk,strawberries + Execute o código e teste, e vejamos o resultado: - -Strawberry milk shake: milk, strawberries, sugar, vanilla extract, ice cubes - -Strawberry shortcake: milk, flour, baking powder, sugar, salt, unsalted butter, strawberries, whipped cream - -Strawberry milk: milk, strawberries, sugar, vanilla extract - ``` + ```output + No of recipes (for example, 5: 3 + List of ingredients (for example, chicken, potatoes, and carrots: milk,strawberries + + -Strawberry milk shake: milk, strawberries, sugar, vanilla extract, ice cubes + -Strawberry shortcake: milk, flour, baking powder, sugar, salt, unsalted butter, strawberries, whipped cream + -Strawberry milk: milk, strawberries, sugar, vanilla extract + ``` ### Melhore adicionando filtro e lista de compras @@ -433,128 +431,128 @@ Para melhorar ainda mais, queremos adicionar o seguinte: - **Filtrar ingredientes**: Queremos ser capazes de filtrar ingredientes que não gostamos ou somos alérgicos. Para realizar essa alteração, podemos editar nosso prompt existente e adicionar uma condição de filtro ao final, assim: - ```python - filter = input("Filter (for example, vegetarian, vegan, or gluten-free: ") - - prompt = f"Show me {no_recipes} recipes for a dish with the following ingredients: {ingredients}. Per recipe, list all the ingredients used, no {filter}" - ``` - - Acima adicionamos `{filter}` ao final do prompt e também capturamos o valor do filtro do usuário. - - Um exemplo de entrada da execução do programa pode ser assim: - - ```output - No of recipes (for example, 5: 3 - List of ingredients (for example, chicken, potatoes, and carrots: onion,milk - Filter (for example, vegetarian, vegan, or gluten-free: no milk - - 1. French Onion Soup - - Ingredients: - - -1 large onion, sliced - -3 cups beef broth - -1 cup milk - -6 slices french bread - -1/4 cup shredded Parmesan cheese - -1 tablespoon butter - -1 teaspoon dried thyme - -1/4 teaspoon salt - -1/4 teaspoon black pepper - - Instructions: - - 1. In a large pot, sauté onions in butter until golden brown. - 2. Add beef broth, milk, thyme, salt, and pepper. Bring to a boil. - 3. Reduce heat and simmer for 10 minutes. - 4. Place french bread slices on soup bowls. - 5. Ladle soup over bread. - 6. Sprinkle with Parmesan cheese. - - 2. Onion and Potato Soup - - Ingredients: - - -1 large onion, chopped - -2 cups potatoes, diced - -3 cups vegetable broth - -1 cup milk - -1/4 teaspoon black pepper - - Instructions: - - 1. In a large pot, sauté onions in butter until golden brown. - 2. Add potatoes, vegetable broth, milk, and pepper. Bring to a boil. - 3. Reduce heat and simmer for 10 minutes. - 4. Serve hot. - - 3. Creamy Onion Soup - - Ingredients: - - -1 large onion, chopped - -3 cups vegetable broth - -1 cup milk - -1/4 teaspoon black pepper - -1/4 cup all-purpose flour - -1/2 cup shredded Parmesan cheese - - Instructions: - - 1. In a large pot, sauté onions in butter until golden brown. - 2. Add vegetable broth, milk, and pepper. Bring to a boil. - 3. Reduce heat and simmer for 10 minutes. - 4. In a small bowl, whisk together flour and Parmesan cheese until smooth. - 5. Add to soup and simmer for an additional 5 minutes, or until soup has thickened. - ``` - - Como você pode ver, todas as receitas com leite foram filtradas. Mas, se você for intolerante à lactose, talvez queira filtrar também as receitas com queijo, então é preciso ser claro. + ```python + filter = input("Filter (for example, vegetarian, vegan, or gluten-free: ") + + prompt = f"Show me {no_recipes} recipes for a dish with the following ingredients: {ingredients}. Per recipe, list all the ingredients used, no {filter}" + ``` + + Acima adicionamos `{filter}` ao final do prompt e também capturamos o valor do filtro do usuário. + + Um exemplo de entrada da execução do programa pode ser assim: + + ```output + No of recipes (for example, 5: 3 + List of ingredients (for example, chicken, potatoes, and carrots: onion,milk + Filter (for example, vegetarian, vegan, or gluten-free: no milk + + 1. French Onion Soup + + Ingredients: + + -1 large onion, sliced + -3 cups beef broth + -1 cup milk + -6 slices french bread + -1/4 cup shredded Parmesan cheese + -1 tablespoon butter + -1 teaspoon dried thyme + -1/4 teaspoon salt + -1/4 teaspoon black pepper + + Instructions: + + 1. In a large pot, sauté onions in butter until golden brown. + 2. Add beef broth, milk, thyme, salt, and pepper. Bring to a boil. + 3. Reduce heat and simmer for 10 minutes. + 4. Place french bread slices on soup bowls. + 5. Ladle soup over bread. + 6. Sprinkle with Parmesan cheese. + + 2. Onion and Potato Soup + + Ingredients: + + -1 large onion, chopped + -2 cups potatoes, diced + -3 cups vegetable broth + -1 cup milk + -1/4 teaspoon black pepper + + Instructions: + + 1. In a large pot, sauté onions in butter until golden brown. + 2. Add potatoes, vegetable broth, milk, and pepper. Bring to a boil. + 3. Reduce heat and simmer for 10 minutes. + 4. Serve hot. + + 3. Creamy Onion Soup + + Ingredients: + + -1 large onion, chopped + -3 cups vegetable broth + -1 cup milk + -1/4 teaspoon black pepper + -1/4 cup all-purpose flour + -1/2 cup shredded Parmesan cheese + + Instructions: + + 1. In a large pot, sauté onions in butter until golden brown. + 2. Add vegetable broth, milk, and pepper. Bring to a boil. + 3. Reduce heat and simmer for 10 minutes. + 4. In a small bowl, whisk together flour and Parmesan cheese until smooth. + 5. Add to soup and simmer for an additional 5 minutes, or until soup has thickened. + ``` + + Como você pode ver, todas as receitas com leite foram filtradas. Mas, se você for intolerante à lactose, talvez queira filtrar também as receitas com queijo, então é preciso ser claro. - **Produze uma lista de compras**. Queremos produzir uma lista de compras, considerando o que já temos em casa. - Para esta funcionalidade, poderíamos tentar resolver tudo em um prompt ou poderíamos dividi-lo em dois prompts. Vamos tentar a última abordagem. Aqui estamos sugerindo adicionar um prompt adicional, mas para que isso funcione, precisamos adicionar o resultado do prompt anterior como contexto para o prompt posterior. + Para esta funcionalidade, poderíamos tentar resolver tudo em um prompt ou poderíamos dividi-lo em dois prompts. Vamos tentar a última abordagem. Aqui estamos sugerindo adicionar um prompt adicional, mas para que isso funcione, precisamos adicionar o resultado do prompt anterior como contexto para o prompt posterior. + + ```python + old_prompt_result = completion.choices[0].text + prompt = "Produce a shopping list for the generated recipes and please don't include ingredients that I already have." + + new_prompt = f"{old_prompt_result} {prompt}" + completion = openai.Completion.create(engine=deployment_name, prompt=new_prompt, max_tokens=1200) + + # print response + print("Shopping list:") + print(completion.choices[0].text) + ``` - ```python - old_prompt_result = completion.choices[0].text - prompt = "Produce a shopping list for the generated recipes and please don't include ingredients that I already have." - - new_prompt = f"{old_prompt_result} {prompt}" - completion = openai.Completion.create(engine=deployment_name, prompt=new_prompt, max_tokens=1200) - - # print response - print("Shopping list:") - print(completion.choices[0].text) - ``` + Observe o seguinte: - Observe o seguinte: + 1. Nós estamos criando um novo prompt adicionando o resultado do primeiro prompt ao novo prompt: - 1. Nós estamos criando um novo prompt adicionando o resultado do primeiro prompt ao novo prompt: + ```python + new_prompt = f"{old_prompt_result} {prompt}" + ``` - ```python - new_prompt = f"{old_prompt_result} {prompt}" - ``` + 1. Nós fizemos uma nova requisição, mas também considerando o número de tokens que solicitamos no primeiro prompt, então desta vez dizemos que `max_tokens` é 1200. - 1. Nós fizemos uma nova requisição, mas também considerando o número de tokens que solicitamos no primeiro prompt, então desta vez dizemos que `max_tokens` é 1200. + ```python + completion = openai.Completion.create(engine=deployment_name, prompt=new_prompt, max_tokens=1200) + ``` - ```python - completion = openai.Completion.create(engine=deployment_name, prompt=new_prompt, max_tokens=1200) - ``` + Dando uma olhada neste código, chegamos agora à seguinte saída: - Dando uma olhada neste código, chegamos agora à seguinte saída: + ```output + No of recipes (for example, 5: 2 + List of ingredients (for example, chicken, potatoes, and carrots: apple,flour + Filter (for example, vegetarian, vegan, or gluten-free: sugar - ```output - No of recipes (for example, 5: 2 - List of ingredients (for example, chicken, potatoes, and carrots: apple,flour - Filter (for example, vegetarian, vegan, or gluten-free: sugar - - - Apple and flour pancakes: 1 cup flour, 1/2 tsp baking powder, 1/2 tsp baking soda, 1/4 tsp salt, 1 tbsp sugar, 1 egg, 1 cup buttermilk or sour milk, 1/4 cup melted butter, 1 Granny Smith apple, peeled and grated + - Apple and flour pancakes: 1 cup flour, 1/2 tsp baking powder, 1/2 tsp baking soda, 1/4 tsp salt, 1 tbsp sugar, 1 egg, 1 cup buttermilk or sour milk, 1/4 cup melted butter, 1 Granny Smith apple, peeled and grated - - Apple fritters: 1-1/2 cups flour, 1 tsp baking powder, 1/4 tsp salt, 1/4 tsp baking soda, 1/4 tsp nutmeg, 1/4 tsp cinnamon, 1/4 tsp allspice, 1/4 cup sugar, 1/4 cup vegetable shortening, 1/4 cup milk, 1 egg, 2 cups shredded, peeled apples + - Apple fritters: 1-1/2 cups flour, 1 tsp baking powder, 1/4 tsp salt, 1/4 tsp baking soda, 1/4 tsp nutmeg, 1/4 tsp cinnamon, 1/4 tsp allspice, 1/4 cup sugar, 1/4 cup vegetable shortening, 1/4 cup milk, 1 egg, 2 cups shredded, peeled apples - Shopping list: - - Flour, baking powder, baking soda, salt, sugar, egg, buttermilk, butter, apple, nutmeg, cinnamon, allspice - ``` + Shopping list: + - Flour, baking powder, baking soda, salt, sugar, egg, buttermilk, butter, apple, nutmeg, cinnamon, allspice + ``` ## Melhorando a nossa configuração @@ -562,47 +560,47 @@ O que temos até agora é um código que funciona, mas há algumas alterações - **Separar `secrets` do código**, como a chave da API. Segredos não pertencem ao código e devem ser armazenados em um local seguro. Para separar segredos do código, podemos usar variáveis de ambiente e bibliotecas como `python-dotenv` para carregá-los de um arquivo. Veja como isso ficaria no código: - 1. Crie um arquivo `.env` com o seguinte conteúdo: + 1. Crie um arquivo `.env` com o seguinte conteúdo: - ```bash - OPENAI_API_KEY=sk-... - ``` + ```bash + OPENAI_API_KEY=sk-... + ``` - > Observação: para o Azure, você precisa definir as seguintes variáveis de ambiente: + > Observação: para o Azure, você precisa definir as seguintes variáveis de ambiente: - ```bash - OPENAI_API_TYPE=azure - OPENAI_API_VERSION=2023-05-15 - OPENAI_API_BASE= - ``` + ```bash + OPENAI_API_TYPE=azure + OPENAI_API_VERSION=2023-05-15 + OPENAI_API_BASE= + ``` - No código, você carregaria as variáveis de ambiente assim: + No código, você carregaria as variáveis de ambiente assim: - ```python - from dotenv import load_dotenv + ```python + from dotenv import load_dotenv - load_dotenv() + load_dotenv() - openai.api_key = os.environ["OPENAI_API_KEY"] - ``` + openai.api_key = os.environ["OPENAI_API_KEY"] + ``` - **Uma palavra sobre o comprimento do token**. Devemos considerar quantos tokens precisamos para gerar o texto que queremos. Tokens custam dinheiro, então, sempre que possível, devemos tentar ser econômicos com o número de tokens que usamos. Por exemplo, podemos formular o prompt para que possamos usar menos tokens? - Para alterar os tokens usados, você pode usar o parâmetro `max_tokens`. Por exemplo, se você quiser usar 100 tokens, você faria: + Para alterar os tokens usados, você pode usar o parâmetro `max_tokens`. Por exemplo, se você quiser usar 100 tokens, você faria: - ```python - completion = openai.Completion.create(model="davinci-002", prompt=prompt, max_tokens=100) - ``` + ```python + completion = openai.Completion.create(model="davinci-002", prompt=prompt, max_tokens=100) + ``` - **Experimentando com a temperatura**. A temperatura é algo que não mencionamos até agora. Porém, é um contexto importante para o desempenho de nosso programa. Quanto maior o valor da temperatura, mais aleatório será o resultado. Por outro lado, quanto menor o valor da temperatura, mais previsível será o resultado. Considere se você deseja ou não variação na saída. - Para alterar a temperatura, você pode usar o parâmetro `temperature`. Por exemplo, se você quiser usar uma temperatura de 0.5, você faria: + Para alterar a temperatura, você pode usar o parâmetro `temperature`. Por exemplo, se você quiser usar uma temperatura de 0.5, você faria: - ```python - completion = openai.Completion.create(model="davinci-002", prompt=prompt, temperature=0.5) - ``` + ```python + completion = openai.Completion.create(model="davinci-002", prompt=prompt, temperature=0.5) + ``` - > Observação: quanto mais próximo de 1.0, mais variada será a saída. + > Observação: quanto mais próximo de 1.0, mais variada será a saída. ## Tarefa diff --git a/07-building-chat-applications/README.md b/07-building-chat-applications/README.md index 74aa725a5..42035d681 100644 --- a/07-building-chat-applications/README.md +++ b/07-building-chat-applications/README.md @@ -1,16 +1,16 @@ # Building Generative AI-Powered Chat Applications -[![Building Generative AI-Powered Chat Applications](./images/07-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/Kw4i-tlKMrQ?WT.mc_id=academic-105485-koreyst) +[![Building Generative AI-Powered Chat Applications](./images/07-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=57a31949-67c5-4020-8c85-91e4995589f3?WT.mc_id=academic-105485-koreyst) -> *(Click the image above to view video of this lesson)* +> _(Click the image above to view video of this lesson)_ Now that we've seen how we can build text-generation apps, let's look into chat applications. -Chat applications have become integrated into our daily lives, offering more than just a means of casual conversation. They're integral parts of customer service, technical support, and even sophisticated advisory systems. It's likely that you've gotten some help from a chat application not too long ago. As we integrate more advanced technologies like generative AI into these platforms, the complexity increases and so does the challenges. +Chat applications have become integrated into our daily lives, offering more than just a means of casual conversation. They're integral parts of customer service, technical support, and even sophisticated advisory systems. It's likely that you've gotten some help from a chat application not too long ago. As we integrate more advanced technologies like generative AI into these platforms, the complexity increases and so does the challenges. Some questions we need to be answered are: -- **Building the app**. How do we efficiently build and seamlessly integrate these AI-powered applications for specific use cases? +- **Building the app**. How do we efficiently build and seamlessly integrate these AI-powered applications for specific use cases? - **Monitoring**. Once deployed, how can we monitor and ensure that the applications are operating at the highest level of quality, both in terms of functionality and adhering to the [six principles of responsible AI](https://www.microsoft.com/ai/responsible-ai?WT.mc_id=academic-105485-koreyst)? As we move further into an age defined by automation and seamless human-machine interactions, understanding how generative AI transforms the scope, depth, and adaptability of chat applications becomes essential. This lesson will investigate the aspects of architecture that support these intricate systems, delve into the methodologies for fine-tuning them for domain-specific tasks, and evaluate the metrics and considerations pertinent to ensuring responsible AI deployment. @@ -82,7 +82,7 @@ The above example uses the GPT-3.5 Turbo model to complete the prompt, but notic General UX principles apply to chat applications, but here are some additional considerations that become particularly important due to the machine learning components involved. - **Mechanism for addressing ambiguity**: Generative AI models occasionally generate ambiguous answers. A feature that allows users to ask for clarification can be helpful should they come across this problem. -- **Context retention**: Advanced generative AI models have the ability to remember context within a conversation, which can be a necessary asset to the user experience. Giving users the ability to control and manage context improves the user experience, but introduces the risk of retaining sensitive user information. Considerations for how long this information is stored, such as introducing a retention policy, can balance the need for context against privacy. +- **Context retention**: Advanced generative AI models have the ability to remember context within a conversation, which can be a necessary asset to the user experience. Giving users the ability to control and manage context improves the user experience, but introduces the risk of retaining sensitive user information. Considerations for how long this information is stored, such as introducing a retention policy, can balance the need for context against privacy. - **Personalization**: With the ability to learn and adapt, AI models offer an individualized experience for a user. Tailoring the user experience through features like user profiles not only makes the user feel understood, but it also helps their pursuit of finding specific answers, creating a more efficient and satisfying interaction. One such example of personalization is the "Custom instructions" settings in OpenAI's ChatGPT. It allows you to provide information about yourself that may be important context for your prompts. Here's an example of a custom instruction. @@ -104,7 +104,7 @@ This "profile" prompts ChatGPT to create a lesson plan on linked lists. Notice t ### Accessibility - Whether a user has visual, auditory, motor, or cognitive impairments, a well-designed chat application should be usable by all. The following list breaks down specific features aimed at enhancing accessibility for various user impairments. +Whether a user has visual, auditory, motor, or cognitive impairments, a well-designed chat application should be usable by all. The following list breaks down specific features aimed at enhancing accessibility for various user impairments. - **Features for Visual Impairment**: High contrast themes and resizable text, screen reader compatibility. - **Features for Auditory Impairment**: Text-to-speech and speech-to-text functions, visual cues for audio notifications. @@ -171,14 +171,14 @@ Microsoft's approach to Responsible AI has identified six principles that should | Privacy and Security | AI systems should be secure and respect privacy. | Implement strong encryption and data protection measures. | To safeguard sensitive user data and comply with privacy laws. | | Inclusiveness | AI systems should empower everyone and engage people. | Design UI/UX that is accessible and easy-to-use for diverse audiences. | Ensures a wider range of people can use the application effectively. | | Transparency | AI systems should be understandable. | Provide clear documentation and reasoning for AI responses. | Users are more likely to trust a system if they can understand how decisions are made. | -| Accountability | People should be accountable for AI systems. | Establish a clear process for auditing and improving AI decisions. | Enables ongoing improvement and corrective measures in case of mistakes. | +| Accountability | People should be accountable for AI systems. | Establish a clear process for auditing and improving AI decisions. | Enables ongoing improvement and corrective measures in case of mistakes. | ## Assignment - + See [assignment](./python?WT.mc_id=academic-105485-koreyst) it will take you through a series of exercises from running your first chat prompts, to classifying and summarizing text and more. Notice that the assignments are available in different programming languages! ## Great Work! Continue the Journey After completing this lesson, check out our [Generative AI Learning collection](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) to continue leveling up your Generative AI knowledge! -Head over to Lesson 8 to see how you can start [building search applications](../08-building-search-applications/README.md?WT.mc_id=academic-105485-koreyst)! +Head over to Lesson 8 to see how you can start [building search applications](../08-building-search-applications/README.md?WT.mc_id=academic-105485-koreyst)! diff --git a/07-building-chat-applications/translations/cn/README.md b/07-building-chat-applications/translations/cn/README.md index d7a06f2dc..4ae96a8aa 100644 --- a/07-building-chat-applications/translations/cn/README.md +++ b/07-building-chat-applications/translations/cn/README.md @@ -1,8 +1,8 @@ # 第七章:创建聊天应用 -[![Building Generative AI-Powered Chat Applications](../../images/07-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/Kw4i-tlKMrQ?WT.mc_id=academic-105485-koreyst) +[![Building Generative AI-Powered Chat Applications](../../images/07-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=57a31949-67c5-4020-8c85-91e4995589f3?WT.mc_id=academic-105485-koreyst) -> *(点击该图片看本章导学视频)* +> _(点击该图片看本章导学视频)_ 通过之前的学习,我们了解了如何构建文本生成应用程序,接下来让我们看看聊天应用程序是如何实现的 @@ -44,14 +44,12 @@ 下表概述了主要差异和相似之处,以帮助我们了解它们在数字通信中的独特作用。 - - -| 传统聊天机器人 | 生成式人工智能驱动的人工智能 | -| ------------------------------------- | -------------------------------------- | -| 以任务为中心、基于预定义规则 | 情境感知 | -| 集成到大型的行业系统中 | 可以托管一个或多个聊天机器人 | -| 通过编程实现 | 结合生成式人工智能模型 | -| 专业相对固化的交互 | 能够进行更开放关于专业领域讨论 | +| 传统聊天机器人 | 生成式人工智能驱动的人工智能 | +| ---------------------------- | ------------------------------ | +| 以任务为中心、基于预定义规则 | 情境感知 | +| 集成到大型的行业系统中 | 可以托管一个或多个聊天机器人 | +| 通过编程实现 | 结合生成式人工智能模型 | +| 专业相对固化的交互 | 能够进行更开放关于专业领域讨论 | ### 结合成熟的 SDK 和 API 构建生成式人工智能驱动的人工智能 @@ -110,7 +108,7 @@ AuthenticationError: No API key provided. You can set your API key in code using ### 辅助功能 - 无论用户是否有视觉、听觉、运动或认知障碍,设计良好的聊天应用程序都应该可供所有人使用。 以下列表详细列出了旨在增强各种用户障碍的可访问性的特定功能。 +无论用户是否有视觉、听觉、运动或认知障碍,设计良好的聊天应用程序都应该可供所有人使用。 以下列表详细列出了旨在增强各种用户障碍的可访问性的特定功能。 - **针对视觉障碍的功能**:高对比度主题和可调整大小的文本、屏幕阅读器兼容性。 - **针对听觉障碍的功能**:文本转语音和语音转文本功能、音频通知的视觉提示。 @@ -134,7 +132,7 @@ AuthenticationError: No API key provided. You can set your API key in code using 例如,医疗查询很复杂并且需要大量上下文。 当医疗专业人员诊断患者时,它会基于多种因素,例如生活方式或既往病史,甚至可能依赖最近的医学期刊来验证他们的诊断。 在这种微妙的场景中,通用的人工智能聊天应用程序不可能成为可靠的来源。 -### 场景:医疗应用** +### 场景:医疗应用\*\* 考虑一个聊天应用程序,旨在通过提供治疗指南、药物相互作用或最新研究结果的快速参考来帮助医生。 @@ -149,51 +147,48 @@ AuthenticationError: No API key provided. You can set your API key in code using 本节概述了“高质量”聊天应用程序的标准,其中包括捕获可操作的指标以及遵守负责任地利用人工智能技术的框架。 -### 关键指标 +### 关键指标 为了保持应用程序的高质量性能,跟踪关键指标和注意事项至关重要。 这些测量不仅可以确保应用程序的功能,还可以评估人工智能模型和用户体验的质量。 下面的列表涵盖了需要考虑的基本、人工智能和用户体验指标。 - -| 指标 | 定义 | 开发人员需要 | -| ----------------------------- | ---------------------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------- | -| **正常运行时间** | 衡量应用程序运行和用户访问的时间。 | 您将如何最大限度地减少停机时间? | -| **响应时间** | 应用程序回复用户查询所花费的时间。 | 如何优化查询处理以缩短响应时间? | -| **精确** | 真阳性预测占阳性预测总数的比率 | 您将如何验证模型的精度? | -| **召回率(灵敏度)** | 真阳性预测与实际阳性数量的比率 | 您将如何衡量和提高召回率? | -| **F1 得分** | 精确率和召回率的调和平均值,平衡两者之间的权衡。 | 您的目标 F1 分数是多少? 您将如何平衡精确度和召回率? | -| **困惑** | 衡量模型预测的概率分布与数据实际分布的吻合程度。 | 您将如何最大程度地减少困惑? | -| **用户满意度指标** | 衡量用户对应用程序的感知。 通常通过调查捕获。 | 您多久收集一次用户反馈? 在此基础上你将如何适应? | -| **错误率** | 模型在理解或输出方面出错的比率。 | 您采取了哪些策略来降低错误率? | -| **再培训周期** | 更新模型以纳入新数据和见解的频率。 | 您多久重新训练一次模型? 是什么触发了再培训周期? | -| **异常检测** | 用于识别不符合预期行为的异常模式的工具和技术。 | 您将如何应对异常情况? | +| 指标 | 定义 | 开发人员需要 | +| -------------------- | ------------------------------------------------ | ----------------------------------------------------- | +| **正常运行时间** | 衡量应用程序运行和用户访问的时间。 | 您将如何最大限度地减少停机时间? | +| **响应时间** | 应用程序回复用户查询所花费的时间。 | 如何优化查询处理以缩短响应时间? | +| **精确** | 真阳性预测占阳性预测总数的比率 | 您将如何验证模型的精度? | +| **召回率(灵敏度)** | 真阳性预测与实际阳性数量的比率 | 您将如何衡量和提高召回率? | +| **F1 得分** | 精确率和召回率的调和平均值,平衡两者之间的权衡。 | 您的目标 F1 分数是多少? 您将如何平衡精确度和召回率? | +| **困惑** | 衡量模型预测的概率分布与数据实际分布的吻合程度。 | 您将如何最大程度地减少困惑? | +| **用户满意度指标** | 衡量用户对应用程序的感知。 通常通过调查捕获。 | 您多久收集一次用户反馈? 在此基础上你将如何适应? | +| **错误率** | 模型在理解或输出方面出错的比率。 | 您采取了哪些策略来降低错误率? | +| **再培训周期** | 更新模型以纳入新数据和见解的频率。 | 您多久重新训练一次模型? 是什么触发了再培训周期? | +| **异常检测** | 用于识别不符合预期行为的异常模式的工具和技术。 | 您将如何应对异常情况? | ### 在聊天应用程序中实施负责任的人工智能实践 Microsoft 的 Responsible AI 方法确定了指导 AI 开发和使用的六项原则。 以下是聊天开发人员应该考虑的原则、定义以及聊天开发人员应该考虑的事项以及为什么他们应该认真对待它们。 -| 原则| 微软的定义| 聊天开发人员的注意事项 | 为什么它很重要 | -| ---------------------- | ----------------------------------------------------- | ---------------------------------------------------------------------- | -------------------------------------------------------------------------------------- | -| 公平| 人工智能系统应该公平对待所有人。 | 确保聊天应用程序不会根据用户数据进行歧视。 | 在用户之间建立信任和包容性; 避免法律后果。 | -| 可靠性和安全性| 人工智能系统应该可靠、安全地运行。 | 实施测试和故障保护以最大限度地减少错误和风险。 | 确保用户满意度并防止潜在危害。 | -| 隐私与安全 | 人工智能系统应该安全并尊重隐私。 | 实施强有力的加密和数据保护措施。 | 保护敏感的用户数据并遵守隐私法。 | -| 包容性| 人工智能系统应该赋予每个人力量并吸引人们。 | 设计适合不同受众且易于使用的 UI/UX。 | 确保更广泛的人可以有效地使用该应用程序。 | -| 透明度 | 人工智能系统应该是可以理解的。 | 为人工智能响应提供清晰的文档和推理。 | 如果用户能够理解决策的制定方式,他们就更有可能信任系统。 | -| 问责制 | 人们应该对人工智能系统负责。 | 建立清晰的审核和改进人工智能决策的流程。 | 允许持续改进并在出现错误时采取纠正措施。 | +| 原则 | 微软的定义 | 聊天开发人员的注意事项 | 为什么它很重要 | +| -------------- | ------------------------------------------ | ---------------------------------------------- | -------------------------------------------------------- | +| 公平 | 人工智能系统应该公平对待所有人。 | 确保聊天应用程序不会根据用户数据进行歧视。 | 在用户之间建立信任和包容性; 避免法律后果。 | +| 可靠性和安全性 | 人工智能系统应该可靠、安全地运行。 | 实施测试和故障保护以最大限度地减少错误和风险。 | 确保用户满意度并防止潜在危害。 | +| 隐私与安全 | 人工智能系统应该安全并尊重隐私。 | 实施强有力的加密和数据保护措施。 | 保护敏感的用户数据并遵守隐私法。 | +| 包容性 | 人工智能系统应该赋予每个人力量并吸引人们。 | 设计适合不同受众且易于使用的 UI/UX。 | 确保更广泛的人可以有效地使用该应用程序。 | +| 透明度 | 人工智能系统应该是可以理解的。 | 为人工智能响应提供清晰的文档和推理。 | 如果用户能够理解决策的制定方式,他们就更有可能信任系统。 | +| 问责制 | 人们应该对人工智能系统负责。 | 建立清晰的审核和改进人工智能决策的流程。 | 允许持续改进并在出现错误时采取纠正措施。 | ## 作业 - + 请查看[作业](../../python?WT.mc_id=academic-105485-koreyst),它将引导您完成一系列练习,从运行第一个聊天提示到对文本进行分类和总结等等。 ## Great Work! Continue the Journey Want to learn more about creating chat applications with Generative AI? Go to the [continued learning page](../../../13-continued-learning/README.md?WT.mc_id=academic-105485-koreyst) to find other great resources on this topic. -Head over to Lesson 8 to see how you can start [building search applications](/08-building-search-applications/README.md?WT.mc_id=academic-105485-koreyst)! - +Head over to Lesson 8 to see how you can start [building search applications](/08-building-search-applications/README.md?WT.mc_id=academic-105485-koreyst)! ## 继续学习 想要了解有关创建聊天应用的更多信息? 转至[进阶学习的页面](../../../13-continued-learning/translations/cn/README.md?WT.mc_id=academic-105485-koreyst) 查找有关此主章节的其他学习资源。 -前往第八章,我们将学习[构建搜索应用程序](../../../08-building-search-applications/translations/cn/README.md?WT.mc_id=academic-105485-koreyst) - +前往第八章,我们将学习[构建搜索应用程序](../../../08-building-search-applications/translations/cn/README.md?WT.mc_id=academic-105485-koreyst) diff --git a/07-building-chat-applications/translations/ja-jp/README.md b/07-building-chat-applications/translations/ja-jp/README.md index e62413f79..3837d5c1e 100644 --- a/07-building-chat-applications/translations/ja-jp/README.md +++ b/07-building-chat-applications/translations/ja-jp/README.md @@ -1,8 +1,8 @@ # 生成 AI を利用したチャット・アプリケーションの構築 -[![Building Generative AI-Powered Chat Applications](../..//images/07-lesson-banner.png?WT.mc_id=academic-105485-yoterada)](https://youtu.be/Kw4i-tlKMrQ?WT.mc_id=academic-105485-yoterada) +[![Building Generative AI-Powered Chat Applications](../..//images/07-lesson-banner.png?WT.mc_id=academic-105485-yoterada)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=57a31949-67c5-4020-8c85-91e4995589f3?WT.mc_id=academic-105485-yoterada) -> *(上記の画像をクリックすると、レッスン・ビデオを表示します)* +> _(上記の画像をクリックすると、レッスン・ビデオを表示します)_ テキスト生成アプリの構築方法を見てきたところで、次にチャット・アプリケーションについて考えてみましょう。 @@ -11,7 +11,7 @@ 解答が必要な質問は下記の通りです。 - **アプリの構築** 特定のユースケースに対して、これらの AI を活用したアプリケーションを効率的に構築し、シームレスに統合するにはどうすればよいのでしょうか? -- **モニタリング** 一度デプロイした後、アプリケーションが機能面と[責任のあるAIの六つの原則](https://www.microsoft.com/ai/responsible-ai?WT.mc_id=academic-105485-yoterada)の準拠という観点から、高品質で稼働している状態をどのようにして監視し、検証できるでしょうか? +- **モニタリング** 一度デプロイした後、アプリケーションが機能面と[責任のある AI の六つの原則](https://www.microsoft.com/ai/responsible-ai?WT.mc_id=academic-105485-yoterada)の準拠という観点から、高品質で稼働している状態をどのようにして監視し、検証できるでしょうか? 自動化が進み、人間と機械が連続でやり取りする時代に突入する中で、生成 AI がチャット・アプリケーションの範囲、深さ、そして適応力をどう変化させるのかを理解するのが重要です。このレッスンでは、これらの複雑なシステムを支えるアーキテクチャを検討し、ドメイン(業種・業界)固有のタスクに合わせてそれらを微調整する方法を考えます。さらに、責任ある AI のデプロイを確実にするための指標と検討内容を学びます。 @@ -44,12 +44,12 @@ 下記の表に、デジタルコミュニケーションにおける、各役割を把握する為の主な違いと共通点をまとめています。 -| チャット・ボット | 生成 AI を搭載したチャットアプリケーション | -| ------------------------------------- | -------------------------------------- | -| タスク中心でルールベース | 文脈(コンテキスト)を理解する | -| 大規模システムへの統合が多い | 1つまたは複数のチャット・ボットをホスト可能 | -| プログラムされた機能に限定 | 生成 AI モデルを搭載 | -| 専門的で構造化された対話を行う | オープン・ドメインでの議論が可能 (特定のトピックや領域に限定しない) | +| チャット・ボット | 生成 AI を搭載したチャットアプリケーション | +| ------------------------------ | ------------------------------------------------------------------- | +| タスク中心でルールベース | 文脈(コンテキスト)を理解する | +| 大規模システムへの統合が多い | 1 つまたは複数のチャット・ボットをホスト可能 | +| プログラムされた機能に限定 | 生成 AI モデルを搭載 | +| 専門的で構造化された対話を行う | オープン・ドメインでの議論が可能 (特定のトピックや領域に限定しない) | ### SDK と API を活用した既存機能の利用 @@ -78,12 +78,12 @@ chat_completion = client.chat.completions.create(model="gpt-3.5-turbo", messages 上記の例では、GPT-3.5 Turbo モデルを使用してプロンプトを作成していますが、その前に API キーを設定している点にご注意ください。キーを設定しなかった場合は、下記のエラーが表示されます。 ```output -AuthenticationError: No API key provided. -You can set your API key in code using 'openai.api_key = ', -or you can set the environment variable OPENAI_API_KEY=). -If your API key is stored in a file, you can point the openai module -at it with 'openai.api_key_path = '. -You can generate API keys in the OpenAI web interface. +AuthenticationError: No API key provided. +You can set your API key in code using 'openai.api_key = ', +or you can set the environment variable OPENAI_API_KEY=). +If your API key is stored in a file, you can point the openai module +at it with 'openai.api_key_path = '. +You can generate API keys in the OpenAI web interface. See https://platform.openai.com/account/api-keys for details. ``` @@ -102,11 +102,11 @@ See https://platform.openai.com/account/api-keys for details. > [!TIP] > 訳者追記 > システムの問い> ChatGPT がより適切な回答を提供するために、あなたに関する情報を教えてください? -> -> 利用者の入力> 私は、大規模な上場ソフトウェア企業で上級職に就いています。2008年からテクノロジーの専門家として活動しており、ソフトウェアエンジニアリングの修士号を持っており、熟練したプログラマーです。また、パートタイムでコンピュータサイエンスの教授も務めています。 -> -> システムの問い> ChatGPT にどのように回答してほしいですか? -> +> +> 利用者の入力> 私は、大規模な上場ソフトウェア企業で上級職に就いています。2008 年からテクノロジーの専門家として活動しており、ソフトウェアエンジニアリングの修士号を持っており、熟練したプログラマーです。また、パートタイムでコンピュータサイエンスの教授も務めています。 +> +> システムの問い> ChatGPT にどのように回答してほしいですか? +> > 利用者の入力> ジャスミンと呼んでください。ChatGPT はいかなるトピックに対しても意見を述べます。 この「プロファイル」で、ChatGPT に Linked Lists(連結リスト)の学習計画を作成するように指示しています。ChatGPT は、利用者の過去の経験や経歴に基づいて、より詳細な学習計画が必要だと判断している点にご注意ください。 @@ -115,21 +115,21 @@ See https://platform.openai.com/account/api-keys for details. > [!TIP] > 訳者追記 -> 利用者の入力> 連結リストの学習計画を作成してください -> -> 回答> もちろんです、ジャスミン!あなたのテクノロジーとコンピュータサイエンス教育のバックグラウンドを考慮すると、基本を超えた詳細な学習計画を望まれていると思います。下記に、コンピュータサイエンスにおける連結リストを教えるための学習計画を提案します。 -> -> 学習計画:連結リストの理解と実装 -> +> 利用者の入力> 連結リストの学習計画を作成してください +> +> 回答> もちろんです、ジャスミン!あなたのテクノロジーとコンピュータサイエンス教育のバックグラウンドを考慮すると、基本を超えた詳細な学習計画を望まれていると思います。下記に、コンピュータサイエンスにおける連結リストを教えるための学習計画を提案します。 +> +> 学習計画:連結リストの理解と実装 +> > 所要時間 -> 1.5時間 -> +> 1.5 時間 +> > 目標 > このレッスンの終わりまでに、生徒は連結リストの概念を理解し、プログラミング言語でそれを実装し、連結リストを用いて問題を解決できるようになる。 ### 大規模言語モデル用の Microsoft のシステム・メッセージ・フレームワーク -Microsoftは、大規模言語モデル(LLM)からの回答を生成する際に、効果的なシステムメッセージを作成するための[ガイダンス](https://learn.microsoft.com/azure/ai-services/openai/concepts/system-message#define-the-models-output-format?WT.mc_id=academic-105485-yoterada)を提供しています。これは以下の 4 つの領域に分類できます: +Microsoft は、大規模言語モデル(LLM)からの回答を生成する際に、効果的なシステムメッセージを作成するための[ガイダンス](https://learn.microsoft.com/azure/ai-services/openai/concepts/system-message#define-the-models-output-format?WT.mc_id=academic-105485-yoterada)を提供しています。これは以下の 4 つの領域に分類できます: 1. モデルのプロファイル、機能、制限を定義する 2. モデルの出力形式を定義する @@ -152,9 +152,9 @@ Microsoftは、大規模言語モデル(LLM)からの回答を生成する - **DSL モデルの活用**:DSL はドメイン固有言語を意味します。ドメインの概念やシナリオを理解するために、特定のドメイン向けにトレーニングした DSL モデルを活用できます。 - **ファインチューニングの適用**:ファイン・チューニングとは、特定のデータを用いてモデルをさらに訓練するプロセスです。 -## カスタマイズ : DSLの使用 +## カスタマイズ : DSL の使用 -ドメイン固有言語モデル(DSLモデル)を活用し、専門用語で文脈に即した内容で AI と対話できれば、利用者の関心を高められます。DSLモデルは、特定の分野、業界、または主題に関連するテキストを理解したり生成するために、訓練またはファイン・チューニングされたモデルです。DSL モデルの利用方法には、ゼロから自分で訓練する方法から、SDK や API を通じて既存のモデルを利用する方法まで様々です。また、既存の事前に訓練済みモデルを取得して、それを特定のドメインに適応させるためにファイン・チューニングするのも選択肢の一つです。 +ドメイン固有言語モデル(DSL モデル)を活用し、専門用語で文脈に即した内容で AI と対話できれば、利用者の関心を高められます。DSL モデルは、特定の分野、業界、または主題に関連するテキストを理解したり生成するために、訓練またはファイン・チューニングされたモデルです。DSL モデルの利用方法には、ゼロから自分で訓練する方法から、SDK や API を通じて既存のモデルを利用する方法まで様々です。また、既存の事前に訓練済みモデルを取得して、それを特定のドメインに適応させるためにファイン・チューニングするのも選択肢の一つです。 ## カスタマイズ:ファイン・チューニングの適用 @@ -181,18 +181,18 @@ Microsoftは、大規模言語モデル(LLM)からの回答を生成する アプリケーションを高品質で高いパフォーマンスを維持するためには、重要な指標(メトリクス)と検討事項を把握する必要があります。これらの測定は、アプリケーションの機能性を保証するだけでなく、AI モデルとユーザー・エクスペリエンスの品質を評価するためにも必要です。下記に、基本的なメトリクス、AI に関するメトリクス、ユーザー・エクスペリエンスに関するメトリクスを検討するためのリストを示します。 -| メトリック | 定義 | チャット開発者の検討事項 | -| ----------------------------- | ---------------------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------- | -| **アップタイム** | アプリケーションが稼働し、利用者が利用可能な時間を計測します | ダウンタイムをどのようにして最小限に抑えられますか? | -| **応答時間** | アプリケーションが利用者からの問い合わせに応答するまでの時間 | クエリ処理をどのように最適化し、応答時間を改善できますか? | -| **Precision(精度)** | 真陽性予測の割合を、陽性予測の総数で割ったもの 

(訳者による追記:モデルで提供される正しい出力の数 (真陽性) と、陽性サンプルの総数 (真陽性+偽陽性) を比較 [ご参考:What Is a Confusion Matrix in Machine Learning?](https://plat.ai/blog/confusion-matrix-in-machine-learning/)) | モデルの精度をどのように確認しますか? | -| **Recall (再現率) (Sensitivity)** | 真陽性予測の数を実際の真の数で割った比率

(訳者による追記:真の値の総数に対する、正しく分類された陽性サンプル(TP)の数の比率として計算 [ご参考:What Is a Confusion Matrix in Machine Learning?](https://plat.ai/blog/confusion-matrix-in-machine-learning/)) | Recall をどのように測定し、改善しますか? | -| **F1 スコア** | 精度と再現率の調和平均で、両者のバランスを示します

(訳者による追記:
F1 = 2*((Precision*Recall)/(Precision+Recall))
FスコアまたはFメジャーは、テストの精度の尺度です。これは、テストの精度と再現率から計算され、精度は真陽性の結果の数を、正しく識別されなかったものを含むすべての陽性結果の数で割った値であり、再現率は、真陽性の結果の数を、陽性として識別されたすべてのサンプルの数で割った値です。) | 目指すF1スコアは何ですか?精度と再現率のバランスをどのように調整しますか? | -| **Perplexity** | モデルが予測した確率分布が、データの実際の分布とどれだけ一致しているかを計測します。

(訳者追記:
[ご参考: Wikipedia パープレキシティ](https://ja.wikipedia.org/wiki/%E3%83%91%E3%83%BC%E3%83%97%E3%83%AC%E3%82%AD%E3%82%B7%E3%83%86%E3%82%A3)) | Perplexity をどのように最小限に抑えますか? | -| **利用者満足度指標** | アプリケーションに対する利用者の評価を計測します。主にアンケートを通じて収集します | ユーザーフィードバックはどのくらいの頻度で収集しますか?それに基づいてどのように改善策を立てますか? | -| **エラー発生率** | モデルが理解や出力で間違いを犯す割合 | エラー発生率を減らすための戦略は何ですか? | -| **再トレーニングサイクル** | 新しいデータや知見を取り入れてモデルを更新する頻度 | どのくらいの頻度でモデルを再トレーニングしますか?再トレーニングサイクルを開始するトリガーは何ですか? | -| **異常検出** | 期待する動作に適合しない異常なパターンを識別するためのツールと手法 | 異常が発生した際にどのように対応しますか? | +| メトリック | 定義 | チャット開発者の検討事項 | +| ----------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ---------------------------------------------------------------------------------------------------- | +| **アップタイム** | アプリケーションが稼働し、利用者が利用可能な時間を計測します | ダウンタイムをどのようにして最小限に抑えられますか? | +| **応答時間** | アプリケーションが利用者からの問い合わせに応答するまでの時間 | クエリ処理をどのように最適化し、応答時間を改善できますか? | +| **Precision(精度)** | 真陽性予測の割合を、陽性予測の総数で割ったもの 

(訳者による追記:モデルで提供される正しい出力の数 (真陽性) と、陽性サンプルの総数 (真陽性+偽陽性) を比較 [ご参考:What Is a Confusion Matrix in Machine Learning?](https://plat.ai/blog/confusion-matrix-in-machine-learning/)) | モデルの精度をどのように確認しますか? | +| **Recall (再現率) (Sensitivity)** | 真陽性予測の数を実際の真の数で割った比率

(訳者による追記:真の値の総数に対する、正しく分類された陽性サンプル(TP)の数の比率として計算 [ご参考:What Is a Confusion Matrix in Machine Learning?](https://plat.ai/blog/confusion-matrix-in-machine-learning/)) |   Recall をどのように測定し、改善しますか? | +| **F1 スコア** | 精度と再現率の調和平均で、両者のバランスを示します

(訳者による追記:
F1 = 2*((Precision*Recall)/(Precision+Recall))
F スコアまたは F メジャーは、テストの精度の尺度です。これは、テストの精度と再現率から計算され、精度は真陽性の結果の数を、正しく識別されなかったものを含むすべての陽性結果の数で割った値であり、再現率は、真陽性の結果の数を、陽性として識別されたすべてのサンプルの数で割った値です。) | 目指す F1 スコアは何ですか?精度と再現率のバランスをどのように調整しますか? | +| **Perplexity** | モデルが予測した確率分布が、データの実際の分布とどれだけ一致しているかを計測します。

(訳者追記:
[ご参考: Wikipedia パープレキシティ](https://ja.wikipedia.org/wiki/%E3%83%91%E3%83%BC%E3%83%97%E3%83%AC%E3%82%AD%E3%82%B7%E3%83%86%E3%82%A3)) | Perplexity をどのように最小限に抑えますか? | +| **利用者満足度指標** | アプリケーションに対する利用者の評価を計測します。主にアンケートを通じて収集します | ユーザーフィードバックはどのくらいの頻度で収集しますか?それに基づいてどのように改善策を立てますか? | +| **エラー発生率** | モデルが理解や出力で間違いを犯す割合 | エラー発生率を減らすための戦略は何ですか? | +| **再トレーニングサイクル** | 新しいデータや知見を取り入れてモデルを更新する頻度 | どのくらいの頻度でモデルを再トレーニングしますか?再トレーニングサイクルを開始するトリガーは何ですか? | +| **異常検出** | 期待する動作に適合しない異常なパターンを識別するためのツールと手法 | 異常が発生した際にどのように対応しますか? | > [!TIP] > 訳者追記:Precision(精度)、Recall (再現率)、F1 スコア、Perplexity は、機械学習の分野でよく使われるメトリクスです。これらのメトリクスの詳細については、[こちら](https://docs.microsoft.com/azure/machine-learning/how-to-understand-automated-ml#classification-metrics?WT.mc_id=academic-105485-yoterada)をご覧ください。 @@ -202,14 +202,14 @@ Microsoftは、大規模言語モデル(LLM)からの回答を生成する Microsoft の責任ある AI への取り組みでは、AI の開発と利用を導く 6 つの原則を定義しています。下記に、それらの原則、定義、及びチャット開発者が検討べき事項と、その重要性について説明します。 -| 原則 | Microsoftの定義 | チャット開発者の検討事項 | 重要な理由 | -| ---------------------- | ----------------------------------------------------- | ---------------------------------------------------------------------- | -------------------------------------------------------------------------------------- | -| 公平性 | AI システムは全ての人を公平に扱うべきです | 利用者データに基づく差別がないようにチャットアプリケーションを設計します | 利用者間で信頼と包括性を築き、法的な問題を避けるためです | -| 信頼性と安全性 |AI システムは確実かつ安全に動作するべきです | エラーやリスクを最小限に抑えるためにテストとフェイルセーフを実装します | 利用者の満足度を確保し、潜在的な危険を防ぐためです | -| プライバシーとセキュリティ | AI システムは安全であり、プライバシーを尊重すべきです | 利用者データを保護するために強力な暗号化とデータ保護対策を実装します | 利用者の機密データを保護し、プライバシー法に準拠するためです | -| 包括性 | AI システムは全ての人に力を与え、人々を惹きつけるものでなければなりません | 多様な利用者がアクセスしやすく使いやすい UI/UX を設計します | 多くの人がアプリケーションを効果的に利用できるようにするためです | -| 透明性 | AI システムは理解しやすいように実装すべきです | AI の回答について明確なドキュメントと理由を提供します | 利用者は、意思決定がどのように行われるかを理解できれば、システムを信頼する可能性が高まります | -| 説明責任 | 人間は AI システムに対して説明責任を負うべきです | AI の意思決定をチェックし、改善するための明確なプロセスを確立する | ミスが発生した際、継続的な改善と修正措置を可能にするためです | +| 原則 | Microsoft の定義 | チャット開発者の検討事項 | 重要な理由 | +| -------------------------- | ------------------------------------------------------------------------- | ------------------------------------------------------------------------ | -------------------------------------------------------------------------------------------- | +| 公平性 | AI システムは全ての人を公平に扱うべきです | 利用者データに基づく差別がないようにチャットアプリケーションを設計します | 利用者間で信頼と包括性を築き、法的な問題を避けるためです | +| 信頼性と安全性 | AI システムは確実かつ安全に動作するべきです | エラーやリスクを最小限に抑えるためにテストとフェイルセーフを実装します | 利用者の満足度を確保し、潜在的な危険を防ぐためです | +| プライバシーとセキュリティ | AI システムは安全であり、プライバシーを尊重すべきです | 利用者データを保護するために強力な暗号化とデータ保護対策を実装します | 利用者の機密データを保護し、プライバシー法に準拠するためです | +| 包括性 | AI システムは全ての人に力を与え、人々を惹きつけるものでなければなりません | 多様な利用者がアクセスしやすく使いやすい UI/UX を設計します | 多くの人がアプリケーションを効果的に利用できるようにするためです | +| 透明性 | AI システムは理解しやすいように実装すべきです | AI の回答について明確なドキュメントと理由を提供します | 利用者は、意思決定がどのように行われるかを理解できれば、システムを信頼する可能性が高まります | +| 説明責任 | 人間は AI システムに対して説明責任を負うべきです | AI の意思決定をチェックし、改善するための明確なプロセスを確立する | ミスが発生した際、継続的な改善と修正措置を可能にするためです | ## 課題 diff --git a/07-building-chat-applications/translations/pt-br/README.md b/07-building-chat-applications/translations/pt-br/README.md index 6dc55bc0b..98e7eafbe 100644 --- a/07-building-chat-applications/translations/pt-br/README.md +++ b/07-building-chat-applications/translations/pt-br/README.md @@ -2,9 +2,9 @@ ## Criando Aplicações Chat com IA Generativa -[![Building Generative AI-Powered Chat Applications](../../images/07-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/Kw4i-tlKMrQ?WT.mc_id=academic-105485-koreyst) +[![Building Generative AI-Powered Chat Applications](../../images/07-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=57a31949-67c5-4020-8c85-91e4995589f3?WT.mc_id=academic-105485-koreyst) -> *(Clique na imagem acima para ver o vídeo da lição)* +> _(Clique na imagem acima para ver o vídeo da lição)_ Agora que vimos como podemos criar aplicativos de geração de texto, vamos dar uma olhada nos aplicativos de bate-papo. @@ -47,12 +47,12 @@ Antes de nos aprofundarmos na criação de aplicativos de chat, vamos comparar ' A tabela abaixo destaca as principais diferenças e semelhanças para nos ajudar a entender seus papéis únicos na comunicação digital. -| Chatbot | Aplicação de Chat Impulsionada por IA Generativa | -| ------------------------------------- | ------------------------------------------------- | -| Focado em tarefas e baseado em regras | Consciente do contexto | -| Frequentemente integrado a sistemas maiores | Pode hospedar um ou vários chatbots | -| Limitado a funções programadas | Incorpora modelos de IA generativa | -| Interações especializadas e estruturadas | Capaz de discussões em domínio aberto | +| Chatbot | Aplicação de Chat Impulsionada por IA Generativa | +| ------------------------------------------- | ------------------------------------------------ | +| Focado em tarefas e baseado em regras | Consciente do contexto | +| Frequentemente integrado a sistemas maiores | Pode hospedar um ou vários chatbots | +| Limitado a funções programadas | Incorpora modelos de IA generativa | +| Interações especializadas e estruturadas | Capaz de discussões em domínio aberto | ### Alavancando funcionalidades pré-criadas com SDKs e APIs @@ -159,38 +159,39 @@ Esta seção delineia os critérios para aplicações de chat consideradas "de a Para manter o desempenho de alta qualidade de uma aplicação, é essencial acompanhar métricas-chave e considerações. Essas medições não apenas garantem a funcionalidade da aplicação, mas também avaliam a qualidade do modelo de IA e a experiência do usuário. Abaixo está uma lista que abrange métricas básicas, de IA e de experiência do usuário a serem consideradas. -| Métrica | Definição | Considerações para o Desenvolvedor de Chat | -| ----------------------------- | ---------------------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------- | -| **Tempo de Atividade (Uptime)**| Mede o tempo em que a aplicação está operacional e acessível pelos usuários. | Como você irá minimizar o tempo de inatividade? | -| **Tempo de Resposta** | O tempo que a aplicação leva para responder a uma consulta do usuário. | Como você pode otimizar o processamento de consultas para melhorar o tempo de resposta? | -| **Precisão** | A razão entre previsões verdadeiras positivas e o número total de previsões positivas. | Como você irá validar a precisão do seu modelo? | -| **Revocação (Sensibilidade)** | A razão entre previsões verdadeiras positivas e o número real de positivos. | Como você irá medir e melhorar a revocação? | -| **Escore F1** | A média harmônica de precisão e revocação, que equilibra o compromisso entre ambos. | Qual é o seu Escore F1 alvo? Como você irá equilibrar precisão e revocação? | -| **Perplexidade** | Mede quão bem a distribuição de probabilidade prevista pelo modelo se alinha com a distribuição real dos dados. | Como você irá minimizar a perplexidade? | -| **Métricas de Satisfação do Usuário** | Mede a percepção do usuário em relação à aplicação. Frequentemente capturado por meio de pesquisas. | Com que frequência você irá coletar feedback do usuário? Como você irá se adaptar a isso? | -| **Taxa de Erro** | A taxa na qual o modelo comete erros na compreensão ou saída. | Quais estratégias você tem para reduzir as taxas de erro? | -| **Ciclos de Retreinamento** | A frequência com que o modelo é atualizado para incorporar novos dados e insights. | Com que frequência você irá retrainer o modelo? O que desencadeia um ciclo de retratamento? | -| **Detecção de Anomalias** | Ferramentas e técnicas para identificar padrões incomuns que não seguem o comportamento esperado. | Como você irá responder a anomalias? | - | +| Métrica | Definição | Considerações para o Desenvolvedor de Chat | +| ------------------------------------- | --------------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------- | +| **Tempo de Atividade (Uptime)** | Mede o tempo em que a aplicação está operacional e acessível pelos usuários. | Como você irá minimizar o tempo de inatividade? | +| **Tempo de Resposta** | O tempo que a aplicação leva para responder a uma consulta do usuário. | Como você pode otimizar o processamento de consultas para melhorar o tempo de resposta? | +| **Precisão** | A razão entre previsões verdadeiras positivas e o número total de previsões positivas. | Como você irá validar a precisão do seu modelo? | +| **Revocação (Sensibilidade)** | A razão entre previsões verdadeiras positivas e o número real de positivos. | Como você irá medir e melhorar a revocação? | +| **Escore F1** | A média harmônica de precisão e revocação, que equilibra o compromisso entre ambos. | Qual é o seu Escore F1 alvo? Como você irá equilibrar precisão e revocação? | +| **Perplexidade** | Mede quão bem a distribuição de probabilidade prevista pelo modelo se alinha com a distribuição real dos dados. | Como você irá minimizar a perplexidade? | +| **Métricas de Satisfação do Usuário** | Mede a percepção do usuário em relação à aplicação. Frequentemente capturado por meio de pesquisas. | Com que frequência você irá coletar feedback do usuário? Como você irá se adaptar a isso? | +| **Taxa de Erro** | A taxa na qual o modelo comete erros na compreensão ou saída. | Quais estratégias você tem para reduzir as taxas de erro? | +| **Ciclos de Retreinamento** | A frequência com que o modelo é atualizado para incorporar novos dados e insights. | Com que frequência você irá retrainer o modelo? O que desencadeia um ciclo de retratamento? | +| **Detecção de Anomalias** | Ferramentas e técnicas para identificar padrões incomuns que não seguem o comportamento esperado. | Como você irá responder a anomalias? | +| | + ### Implementando Práticas de IA Responsável em Aplicações de Chat A abordagem da Microsoft para a IA Responsável identificou seis princípios que devem orientar o desenvolvimento e uso de IA. Abaixo estão os princípios, suas definições e o que um desenvolvedor de chat deve considerar e por que deve levar isso a sério. -| Princípios | Definição pela Microsoft | Considerações para o Desenvolvedor de Chat | Por que é Importante | -| ----------------------- | ------------------------------------------------------ | ------------------------------------------------------------------------- | -------------------------------------------------------------------------------------- | -| Justiça | Sistemas de IA devem tratar todas as pessoas de forma justa. | Garantir que a aplicação de chat não discrimine com base nos dados do usuário. | Construir confiança e inclusividade entre os usuários; evita ramificações legais. | -| Confiabilidade e Segurança | Sistemas de IA devem ter desempenho confiável e seguro. | Implementar testes e mecanismos de segurança para minimizar erros e riscos. | Garante satisfação do usuário e evita possíveis danos. | -| Privacidade e Segurança | Sistemas de IA devem ser seguros e respeitar a privacidade. | Implementar criptografia forte e medidas de proteção de dados. | Proteger dados sensíveis do usuário e cumprir leis de privacidade. | -| Inclusividade | Sistemas de IA devem capacitar todos e envolver as pessoas. | Projetar uma interface acessível e fácil de usar para diversos públicos. | Garante que uma variedade maior de pessoas possa usar a aplicação de forma eficaz. | -| Transparência | Sistemas de IA devem ser compreensíveis. | Fornecer documentação clara e justificativa para as respostas da IA. | Os usuários têm mais probabilidade de confiar em um sistema se puderem entender como são tomadas as decisões. | -| Responsabilidade | As pessoas devem ser responsáveis pelos sistemas de IA. | Estabelecer um processo claro para auditoria e melhoria das decisões de IA. | Possibilita melhorias contínuas e medidas corretivas em caso de erros. | +| Princípios | Definição pela Microsoft | Considerações para o Desenvolvedor de Chat | Por que é Importante | +| -------------------------- | ------------------------------------------------------------ | ------------------------------------------------------------------------------ | ------------------------------------------------------------------------------------------------------------- | +| Justiça | Sistemas de IA devem tratar todas as pessoas de forma justa. | Garantir que a aplicação de chat não discrimine com base nos dados do usuário. | Construir confiança e inclusividade entre os usuários; evita ramificações legais. | +| Confiabilidade e Segurança | Sistemas de IA devem ter desempenho confiável e seguro. | Implementar testes e mecanismos de segurança para minimizar erros e riscos. | Garante satisfação do usuário e evita possíveis danos. | +| Privacidade e Segurança | Sistemas de IA devem ser seguros e respeitar a privacidade. | Implementar criptografia forte e medidas de proteção de dados. | Proteger dados sensíveis do usuário e cumprir leis de privacidade. | +| Inclusividade | Sistemas de IA devem capacitar todos e envolver as pessoas. | Projetar uma interface acessível e fácil de usar para diversos públicos. | Garante que uma variedade maior de pessoas possa usar a aplicação de forma eficaz. | +| Transparência | Sistemas de IA devem ser compreensíveis. | Fornecer documentação clara e justificativa para as respostas da IA. | Os usuários têm mais probabilidade de confiar em um sistema se puderem entender como são tomadas as decisões. | +| Responsabilidade | As pessoas devem ser responsáveis pelos sistemas de IA. | Estabelecer um processo claro para auditoria e melhoria das decisões de IA. | Possibilita melhorias contínuas e medidas corretivas em caso de erros. | ## Tarefa - + Veja a [tarefa](../../python?WT.mc_id=academic-105485-koreyst) que o levará por uma série de exercícios, desde a execução de seus primeiros prompts de chat até a classificação e resumo de texto e muito mais. ## Excelente Trabalho! Continue no Aprendizado! Depois de concluir esta lição, confira nossa [coleção de aprendizado de IA generativa](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) para continuar a aprimorar seus conhecimentos sobre IA generativa! -Agora, vamos seguir para a Lição 8 para ver como você pode começar a [criar aplicativos de pesquisa](../../../08-building-search-applications/translations/pt-br/README.md?WT.mc_id=academic-105485-koreyst)! \ No newline at end of file +Agora, vamos seguir para a Lição 8 para ver como você pode começar a [criar aplicativos de pesquisa](../../../08-building-search-applications/translations/pt-br/README.md?WT.mc_id=academic-105485-koreyst)! diff --git a/10-building-low-code-ai-applications/README.md b/10-building-low-code-ai-applications/README.md index 5bdc16a38..dd73aa51d 100644 --- a/10-building-low-code-ai-applications/README.md +++ b/10-building-low-code-ai-applications/README.md @@ -1,8 +1,8 @@ # Building Low Code AI Applications -[![Building Low Code AI Applications](./images/10-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/XX8491SAF44?WT.mc_id=academic-105485-koreyst) +[![Building Low Code AI Applications](./images/10-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=170c6cb7-3c96-40f2-a354-b6f0ec523de7?WT.mc_id=academic-105485-koreyst) -> *(Click the image above to view video of this lesson)* +> _(Click the image above to view video of this lesson)_ ## Introduction @@ -43,7 +43,7 @@ The tools and technologies that you will use in this lesson are: ## Generative AI in Power Platform -Enhancing low-code development and application with generative AI is a key focus area for Power Platform. The goal is to enable everyone to build AI-powered apps, sites, dashboards and automate processes with AI, *without requiring any data science expertise*. This goal is achieved by integrating generative AI into the low-code development experience in Power Platform in the form of Copilot and AI Builder. +Enhancing low-code development and application with generative AI is a key focus area for Power Platform. The goal is to enable everyone to build AI-powered apps, sites, dashboards and automate processes with AI, _without requiring any data science expertise_. This goal is achieved by integrating generative AI into the low-code development experience in Power Platform in the form of Copilot and AI Builder. ### How does this work? @@ -81,17 +81,17 @@ You will build the app using Copilot in Power Apps following the steps below: 1. Navigate to the [Power Apps](https://make.powerapps.com?WT.mc_id=academic-105485-koreyst) home screen. -1. Use the text area on the home screen to describe the app you want to build. For example, ***I want to build an app to track and manage student assignments***. Click on the **Send** button to send the prompt to the AI Copilot. +1. Use the text area on the home screen to describe the app you want to build. For example, **_I want to build an app to track and manage student assignments_**. Click on the **Send** button to send the prompt to the AI Copilot. - ![Describe the app you want to build](./images/copilot-chat-prompt-powerapps.png?WT.mc_id=academic-105485-koreyst) +![Describe the app you want to build](./images/copilot-chat-prompt-powerapps.png?WT.mc_id=academic-105485-koreyst) 1. The AI Copilot will suggest a Dataverse Table with the fields you need to store the data you want to track and some sample data. You can then customize the table to meet your needs using the AI Copilot assistant feature through conversational steps. - > **Important**: Dataverse is the underlying data platform for Power Platform. It is a low-code data platform for storing the app's data. It is a fully managed service that securely stores data in the Microsoft Cloud and is provisioned within your Power Platform environment. It comes with built-in data governance capabilities, such as data classification, data lineage, fine-grained access control, and more. You can learn more about Dataverse [here](https://docs.microsoft.com/powerapps/maker/data-platform/data-platform-intro?WT.mc_id=academic-109639-somelezediko). + > **Important**: Dataverse is the underlying data platform for Power Platform. It is a low-code data platform for storing the app's data. It is a fully managed service that securely stores data in the Microsoft Cloud and is provisioned within your Power Platform environment. It comes with built-in data governance capabilities, such as data classification, data lineage, fine-grained access control, and more. You can learn more about Dataverse [here](https://docs.microsoft.com/powerapps/maker/data-platform/data-platform-intro?WT.mc_id=academic-109639-somelezediko). ![Suggested fields in your new table](./images/copilot-dataverse-table-powerapps.png?WT.mc_id=academic-105485-koreyst) -1. Educators want to send emails to the students who have submitted their assignments to keep them updated on the progress of their assignments. You can use Copilot to add a new field to the table to store the student email. For example, you can use the following prompt to add a new field to the table: ***I want to add a column to store student email***. Click on the **Send** button to send the prompt to the AI Copilot. +1. Educators want to send emails to the students who have submitted their assignments to keep them updated on the progress of their assignments. You can use Copilot to add a new field to the table to store the student email. For example, you can use the following prompt to add a new field to the table: **_I want to add a column to store student email_**. Click on the **Send** button to send the prompt to the AI Copilot. ![Adding a new field](./images/copilot-new-column.png?WT.mc_id=academic-105485-koreyst) @@ -101,7 +101,7 @@ You will build the app using Copilot in Power Apps following the steps below: 1. The AI Copilot will generate a responsive Canvas app based on your description. You can then customize the app to meet your needs. -1. For educators to send emails to students, you can use Copilot to add a new screen to the app. For example, you can use the following prompt to add a new screen to the app: ***I want to add a screen to send emails to students***. Click on the **Send** button to send the prompt to the AI Copilot. +1. For educators to send emails to students, you can use Copilot to add a new screen to the app. For example, you can use the following prompt to add a new screen to the app: **_I want to add a screen to send emails to students_**. Click on the **Send** button to send the prompt to the AI Copilot. ![Adding a new screen via a prompt instruction](./images/copilot-new-screen.png?WT.mc_id=academic-105485-koreyst) @@ -131,7 +131,7 @@ Why should we use Dataverse for our startup? The standard and custom tables with Now that you know what Dataverse is and why you should use it, let's look at how you can use Copilot to create a table in Dataverse to meet the requirements of our finance team. ->**Note** : You will use this table in the next section to build an automation that will extract all the invoice information and store it in the table. +> **Note** : You will use this table in the next section to build an automation that will extract all the invoice information and store it in the table. To create a table in Dataverse using Copilot, follow the steps below: @@ -141,7 +141,7 @@ To create a table in Dataverse using Copilot, follow the steps below: ![Select new table](./images/describe-new-table.png?WT.mc_id=academic-105485-koreyst) -1. On the **Describe the new Table** screen, use the text area to describe the table you want to create. For example, ***I want to create a table to store invoice information***. Click on the **Send** button to send the prompt to the AI Copilot. +1. On the **Describe the new Table** screen, use the text area to describe the table you want to create. For example, **_I want to create a table to store invoice information_**. Click on the **Send** button to send the prompt to the AI Copilot. ![Describe the table](./images/copilot-chat-prompt-dataverse.png?WT.mc_id=academic-105485-koreyst) @@ -149,7 +149,7 @@ To create a table in Dataverse using Copilot, follow the steps below: ![Suggested Dataverse table](./images/copilot-dataverse-table.png?WT.mc_id=academic-105485-koreyst) -1. The finance team wants to send an email to the supplier to update them with the current status of their invoice. You can use Copilot to add a new field to the table to store the supplier email. For example, you can use the following prompt to add a new field to the table: ***I want to add a column to store supplier email***. Click on the **Send** button to send the prompt to the AI Copilot. +1. The finance team wants to send an email to the supplier to update them with the current status of their invoice. You can use Copilot to add a new field to the table to store the supplier email. For example, you can use the following prompt to add a new field to the table: **_I want to add a column to store supplier email_**. Click on the **Send** button to send the prompt to the AI Copilot. 1. The AI Copilot will generate a new field and you can then customize the field to meet your needs. @@ -188,9 +188,9 @@ To build a workflow that will help the finance team process invoices using the I 1. Navigate to the [Power Automate](https://make.powerautomate.com?WT.mc_id=academic-105485-koreyst) home screen. -2. Use the text area on the home screen to describe the workflow you want to build. For example, ***Process an invoice when it arrives in my mailbox***. Click on the **Send** button to send the prompt to the AI Copilot. +2. Use the text area on the home screen to describe the workflow you want to build. For example, **_Process an invoice when it arrives in my mailbox_**. Click on the **Send** button to send the prompt to the AI Copilot. - ![Copilot power automate](./images/copilot-chat-prompt-powerautomate.png?WT.mc_id=academic-105485-koreyst) + ![Copilot power automate](./images/copilot-chat-prompt-powerautomate.png?WT.mc_id=academic-105485-koreyst) 3. The AI Copilot will suggest the actions you need to perform the task you want to automate. You can click on the **Next** button to go through the next steps. @@ -204,7 +204,7 @@ To build a workflow that will help the finance team process invoices using the I 8. Remove the **Condition** action from the flow because you will not be using it. It should look like the following screenshot: - ![power automate, remove actions](./images/powerautomate-remove-actions.png?WT.mc_id=academic-105485-koreyst) + ![power automate, remove actions](./images/powerautomate-remove-actions.png?WT.mc_id=academic-105485-koreyst) 9. Click on the **Add an action** button and search for **Dataverse**. Select the **Add a new row** action. diff --git a/10-building-low-code-ai-applications/translations/cn/README.md b/10-building-low-code-ai-applications/translations/cn/README.md index efe165fc2..485051852 100644 --- a/10-building-low-code-ai-applications/translations/cn/README.md +++ b/10-building-low-code-ai-applications/translations/cn/README.md @@ -1,8 +1,8 @@ # 第十章:创建低代码的人工智能应用 -[![Building Low Code AI Applications](../../images/10-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/XX8491SAF44?WT.mc_id=academic-105485-koreyst) +[![Building Low Code AI Applications](../../images/10-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=170c6cb7-3c96-40f2-a354-b6f0ec523de7?WT.mc_id=academic-105485-koreyst) -> *(点击该图片看本章导学视频)* +> _(点击该图片看本章导学视频)_ ## 本章概述 @@ -43,7 +43,7 @@ Power Platform 为企业提供了让其团队能够通过直观的低代码或 ## Power Platform 中的生成式 AI -通过生成式 AI 增强低代码开发和应用是 Power Platform 的重点关注领域。 目标是让每个人都能够构建人工智能驱动的应用程序、网站、仪表板并利用人工智能实现流程自动化,*无需任何数据科学专业知识*。 这一目标是通过将生成式 AI 以 Copilot 和 AI Builder 的形式集成到 Power Platform 的低代码开发体验中来实现的。 +通过生成式 AI 增强低代码开发和应用是 Power Platform 的重点关注领域。 目标是让每个人都能够构建人工智能驱动的应用程序、网站、仪表板并利用人工智能实现流程自动化,_无需任何数据科学专业知识_。 这一目标是通过将生成式 AI 以 Copilot 和 AI Builder 的形式集成到 Power Platform 的低代码开发体验中来实现的。 ### 这是如何运作的? @@ -55,17 +55,17 @@ AI Builder 是 Power Platform 中提供的低代码 AI 功能,使您能够使 Copilot 适用于所有 Power Platform 产品:Power Apps、Power Automate、Power BI、Power Pages 和 Power Virtual Agent。 AI Builder 可在 Power Apps 和 Power Automate 中使用。 在本课程中,我们将重点介绍如何在 Power Apps 和 Power Automate 中使用 Copilot 和 AI Builder 为我们的教育初创公司构建解决方案。 -### Power Apps 中的 Copilot +### Power Apps 中的 Copilot 作为 Power Platform 的一部分,Power Apps 提供了一个低代码开发环境,用于构建应用程序来跟踪、管理数据并与数据交互。 它是一套应用程序开发服务,具有可扩展的数据平台以及连接到云服务和本地数据的能力。 Power Apps 允许您构建在浏览器、平板电脑和手机上运行并可以与同事共享的应用程序。 Power Apps 通过简单的界面让用户轻松进行应用程序开发,以便每个业务用户或专业开发人员都可以构建自定义应用程序。 通过 Copilot 的生成式 AI 也增强了应用程序开发体验。 -Power Apps 中的 Copilot AI 助手功能使您能够描述您需要哪种类型的应用程序以及您希望应用程序跟踪、收集或显示哪些信息。 然后,Copilot 根据您的描述生成响应式 Canvas 应用程序。 然后,您可以自定义应用程序以满足您的需求。 AI Copilot 还会生成并建议一个 Dataverse 表,其中包含存储要跟踪的数据和一些示例数据所需的字段。 稍后我们将在本章中了解什么是 Dataverse 以及如何在 Power Apps 中使用它。 然后,您可以通过对话步骤使用 AI Copilot 助手功能自定义表格以满足您的需求。 通过 Power Apps 主屏幕轻松使用此功能。 +Power Apps 中的 Copilot AI 助手功能使您能够描述您需要哪种类型的应用程序以及您希望应用程序跟踪、收集或显示哪些信息。 然后,Copilot 根据您的描述生成响应式 Canvas 应用程序。 然后,您可以自定义应用程序以满足您的需求。 AI Copilot 还会生成并建议一个 Dataverse 表,其中包含存储要跟踪的数据和一些示例数据所需的字段。 稍后我们将在本章中了解什么是 Dataverse 以及如何在 Power Apps 中使用它。 然后,您可以通过对话步骤使用 AI Copilot 助手功能自定义表格以满足您的需求。 通过 Power Apps 主屏幕轻松使用此功能。 -### Power Automate 中的 Copilot +### Power Automate 中的 Copilot 作为 Power Platform 的一部分,Power Automate 允许用户在应用程序和服务之间创建自动化工作流程。 它有助于自动化重复的业务流程,例如通信、数据收集和决策批准。 其简单的界面允许具有各种技术能力的用户(从初学者到经验丰富的开发人员)自动执行工作任务。 通过 Copilot 的生成式 AI 也增强了工作流程开发体验。 -Power Automate 中的 Copilot AI 助手功能使您能够描述您需要哪种流程以及您希望流程执行哪些操作。 然后 Copilot 根据您的描述生成流程。 然后,您可以自定义流程以满足您的需求。 AI Copilot 还会生成并建议您执行想要自动化的任务所需的操作。 我们将在本章中稍后介绍什么是流以及如何在 Power Automate 中使用它们。 然后,您可以通过对话步骤使用 AI Copilot 助手功能自定义操作以满足您的需求。 通过 Power Automate 主屏幕轻松使用此功能。 +Power Automate 中的 Copilot AI 助手功能使您能够描述您需要哪种流程以及您希望流程执行哪些操作。 然后 Copilot 根据您的描述生成流程。 然后,您可以自定义流程以满足您的需求。 AI Copilot 还会生成并建议您执行想要自动化的任务所需的操作。 我们将在本章中稍后介绍什么是流以及如何在 Power Automate 中使用它们。 然后,您可以通过对话步骤使用 AI Copilot 助手功能自定义操作以满足您的需求。 通过 Power Automate 主屏幕轻松使用此功能。 ## 作业:使用 Copilot 管理 “Our Startup” 的学生作业和发票 @@ -81,17 +81,17 @@ Power Automate 中的 Copilot AI 助手功能使您能够描述您需要哪种 1. 导航到 [Power Apps](https://make.powerapps.com?WT.mc_id=academic-105485-koreyst) 主屏幕。 -1. 使用主屏幕上的文本区域描述您要构建的应用程序。 例如,***我想构建一个应用程序来跟踪和管理学生作业***。 单击 **发送** 按钮将提示发送到 AI Copilot。 +1. 使用主屏幕上的文本区域描述您要构建的应用程序。 例如,**_我想构建一个应用程序来跟踪和管理学生作业_**。 单击 **发送** 按钮将提示发送到 AI Copilot。 - ![描述您要构建的应用程序](../../images/copilot-chat-prompt-powerapps.png?WT.mc_id=academic-105485-koreyst) +![描述您要构建的应用程序](../../images/copilot-chat-prompt-powerapps.png?WT.mc_id=academic-105485-koreyst) 1. AI Copilot 将建议一个 Dataverse 表,其中包含存储要跟踪的数据所需的字段以及一些示例数据。 然后,您可以通过对话步骤使用 AI Copilot 助手功能自定义表格以满足您的需求。 - > **重点事项**:Dataverse 是 Power Platform 的底层数据平台。 它是一个用于存储应用程序数据的低代码数据平台。 它是一项完全托管的服务,可将数据安全地存储在 Microsoft 云中,并在您的 Power Platform 环境中进行配置。 它具有内置的数据治理功能,例如数据分类、数据沿袭、细粒度访问控制等。 您可以在[此处](https://docs.microsoft.com/powerapps/maker/data-platform/data-platform-intro?WT.mc_id=academic-109639-somelezediko)了解有关Dataverse的更多信息。 + > **重点事项**:Dataverse 是 Power Platform 的底层数据平台。 它是一个用于存储应用程序数据的低代码数据平台。 它是一项完全托管的服务,可将数据安全地存储在 Microsoft 云中,并在您的 Power Platform 环境中进行配置。 它具有内置的数据治理功能,例如数据分类、数据沿袭、细粒度访问控制等。 您可以在[此处](https://docs.microsoft.com/powerapps/maker/data-platform/data-platform-intro?WT.mc_id=academic-109639-somelezediko)了解有关 Dataverse 的更多信息。 - ![新表中的建议字段](../../images/copilot-dataverse-table-powerapps.png?WT.mc_id=academic-105485-koreyst) + ![新表中的建议字段](../../images/copilot-dataverse-table-powerapps.png?WT.mc_id=academic-105485-koreyst) -2. 教育工作者希望向已提交作业的学生发送电子邮件,以便让他们了解作业的最新进度。 您可以使用 Copilot 向表中添加新字段来存储学生电子邮件。 例如,您可以使用以下提示向表中添加新字段:***我想添加一列来存储学生电子邮件***。 单击 **发送** 按钮将提示发送到 AI Copilot。 +2. 教育工作者希望向已提交作业的学生发送电子邮件,以便让他们了解作业的最新进度。 您可以使用 Copilot 向表中添加新字段来存储学生电子邮件。 例如,您可以使用以下提示向表中添加新字段:**_我想添加一列来存储学生电子邮件_**。 单击 **发送** 按钮将提示发送到 AI Copilot。 ![添加新字段](../../images/copilot-new-column.png?WT.mc_id=academic-105485-koreyst) @@ -101,7 +101,7 @@ Power Automate 中的 Copilot AI 助手功能使您能够描述您需要哪种 1. AI Copilot 将根据您的描述生成响应式 Canvas 应用程序。 然后,您可以自定义应用程序以满足您的需求。 -1. 对于要向学生发送电子邮件的教育工作者,您可以使用 Copilot 向应用程序添加新屏幕。 例如,您可以使用以下提示向应用程序添加新屏幕:***我想添加一个屏幕来向学生发送电子邮件***。 单击 **发送** 按钮将提示发送到 AI Copilot。 +1. 对于要向学生发送电子邮件的教育工作者,您可以使用 Copilot 向应用程序添加新屏幕。 例如,您可以使用以下提示向应用程序添加新屏幕:**_我想添加一个屏幕来向学生发送电子邮件_**。 单击 **发送** 按钮将提示发送到 AI Copilot。 ![Describe the app you want to build](../../images/copilot-new-screen.png?WT.mc_id=academic-105485-koreyst) @@ -113,10 +113,9 @@ Power Automate 中的 Copilot AI 助手功能使您能够描述您需要哪种 > **提升**:你刚刚构建的应用程序是一个良好的开端,但还可以改进。 通过电子邮件功能,教育工作者只能通过键入电子邮件来手动向学生发送电子邮件。 您能否使用 Copilot 构建一个自动化系统,使教育工作者能够在学生提交作业时自动向他们发送电子邮件? 您的提示是正确的提示,您可以 - ### 为 “Our Startup” 构建发票信息表 - “Our Startup” 的财务团队一直在努力跟踪发票。 他们一直使用电子表格来跟踪发票,但随着发票数量的增加,这变得难以管理。 他们要求您构建一个表来帮助他们存储、跟踪和管理收到的发票信息。 该表应用于构建一个自动化程序,提取所有发票信息并将其存储在表中。 该表还应使财务团队能够查看已支付和未支付的发票。 +“Our Startup” 的财务团队一直在努力跟踪发票。 他们一直使用电子表格来跟踪发票,但随着发票数量的增加,这变得难以管理。 他们要求您构建一个表来帮助他们存储、跟踪和管理收到的发票信息。 该表应用于构建一个自动化程序,提取所有发票信息并将其存储在表中。 该表还应使财务团队能够查看已支付和未支付的发票。 Power Platform 有一个名为 Dataverse 的底层数据平台,使您能够存储应用程序和解决方案的数据。 Dataverse 提供了一个低代码数据平台来存储应用程序的数据。 它是一项完全托管的服务,可将数据安全地存储在 Microsoft 云中,并在您的 Power Platform 环境中进行配置。 它具有内置的数据治理功能,例如数据分类、数据沿袭、细粒度访问控制等。 您可以在此处了解[有关 Dataverse 的更多信息](https://docs.microsoft.com/powerapps/maker/data-platform/data-platform-intro?WT.mc_id=academic-109639-somelezediko)。 @@ -132,7 +131,7 @@ Power Platform 有一个名为 Dataverse 的底层数据平台,使您能够存 现在您已经了解了 Dataverse 是什么以及为什么应该使用它,让我们看看如何使用 Copilot 在 Dataverse 中创建表来满足我们财务团队的要求。 ->**注意**:您将在下面内容中使用此表来构建自动化,该自动化将提取所有发票信息并将其存储在表中。 +> **注意**:您将在下面内容中使用此表来构建自动化,该自动化将提取所有发票信息并将其存储在表中。 要使用 Copilot 在 Dataverse 中创建表,请按照以下步骤操作: @@ -142,7 +141,7 @@ Power Platform 有一个名为 Dataverse 的底层数据平台,使您能够存 ![选择新表](../../images/describe-new-table.png?WT.mc_id=academic-105485-koreyst) -1. 在 **描述新表** 屏幕上,使用文本区域描述您要创建的表。 例如,***我想创建一个表来存储发票信息***。 单击 **发送** 按钮将提示发送到 AI Copilot。 +1. 在 **描述新表** 屏幕上,使用文本区域描述您要创建的表。 例如,**_我想创建一个表来存储发票信息_**。 单击 **发送** 按钮将提示发送到 AI Copilot。 ![描述表格](../../images/copilot-chat-prompt-dataverse.png?WT.mc_id=academic-105485-koreyst) @@ -150,7 +149,7 @@ Power Platform 有一个名为 Dataverse 的底层数据平台,使您能够存 ![建议的 Dataverse 表](../../images/copilot-dataverse-table.png?WT.mc_id=academic-105485-koreyst) -1. 财务团队希望向供应商发送电子邮件,以更新其发票的当前状态。 您可以使用 Copilot 向表中添加新字段来存储供应商电子邮件。 例如,您可以使用以下提示向表中添加新字段:***我想添加一列来存储供应商电子邮件***。 单击 **发送** 按钮将提示发送到 AI Copilot。 +1. 财务团队希望向供应商发送电子邮件,以更新其发票的当前状态。 您可以使用 Copilot 向表中添加新字段来存储供应商电子邮件。 例如,您可以使用以下提示向表中添加新字段:**_我想添加一列来存储供应商电子邮件_**。 单击 **发送** 按钮将提示发送到 AI Copilot。 1. AI Copilot 将生成一个新字段,您可以自定义该字段以满足您的需求。 @@ -189,9 +188,9 @@ Power Platform 中提供的一些预构建 AI 模型包括: 1. 导航至 [Power Automate](https://make.powerautomate.com?WT.mc_id=academic-105485-koreyst) 主屏幕。 -2. 使用主屏幕上的文本区域描述您要构建的工作流程。 例如,***当发票到达我的邮箱时对其进行处理***。 单击 **发送** 按钮将提示发送到 AI Copilot。 +2. 使用主屏幕上的文本区域描述您要构建的工作流程。 例如,**_当发票到达我的邮箱时对其进行处理_**。 单击 **发送** 按钮将提示发送到 AI Copilot。 - ![副驾驶电源自动化](../../images/copilot-chat-prompt-powerautomate.png?WT.mc_id=academic-105485-koreyst) + ![副驾驶电源自动化](../../images/copilot-chat-prompt-powerautomate.png?WT.mc_id=academic-105485-koreyst) 3. AI Copilot 将建议您执行想要自动化的任务所需的操作。 您可以单击 **下一步** 按钮来完成后续步骤。 @@ -205,7 +204,7 @@ Power Platform 中提供的一些预构建 AI 模型包括: 8. 从流程中删除 **Condition** 操作,因为您不会使用它。 它应该类似于以下屏幕截图: - ![电源自动化,删除操作](../../images/powerautomate-remove-actions.png?WT.mc_id=academic-105485-koreyst) + ![电源自动化,删除操作](../../images/powerautomate-remove-actions.png?WT.mc_id=academic-105485-koreyst) 9. 单击 **添加操作** 按钮并搜索 **Dataverse**。 选择 **添加新行** 操作。 @@ -213,20 +212,19 @@ Power Platform 中提供的一些预构建 AI 模型包括: 11. 选择您之前创建的 **表**。 例如,您可以选择 **发票信息** 表。 从上一个操作中选择动态内容以填充以下字段: - - ID - - 数量 - - 日期 - - 姓名 - - 状态 - 将 **状态** 设置为 **待处理**。 - - 供应商电子邮件 - 使用 **新电子邮件到达时** 触发器中的 **发件人** 动态内容。 + - ID + - 数量 + - 日期 + - 姓名 + - 状态 - 将 **状态** 设置为 **待处理**。 + - 供应商电子邮件 - 使用 **新电子邮件到达时** 触发器中的 **发件人** 动态内容。 - ![电源自动添加行](../../images/powerautomate-add-row.png?WT.mc_id=academic-105485-koreyst) + ![电源自动添加行](../../images/powerautomate-add-row.png?WT.mc_id=academic-105485-koreyst) 12. 完成流程后,单击 **保存** 按钮保存流程。 然后,您可以通过将带有发票的电子邮件发送到您在触发器中指定的文件夹来测试流程。 > **提升**:你刚刚构建的流程是一个好的开始,现在你需要考虑如何构建一个自动化系统,使我们的财务团队能够向供应商发送电子邮件以更新他们的当前状态 他们的发票。 您的提示:当发票状态发生变化时,流程必须运行。 - ## 在 Power Automate 中使用文本生成 AI 模型 AI Builder 中的使用 GPT AI 模型创建文本使您能够根据提示生成文本,并由 Microsoft Azure OpenAI Service 提供支持。 借助此功能,您可以将 GPT 技术合并到您的应用程序和流程中,以构建各种自动化流程和富有洞察力的应用程序。 diff --git a/10-building-low-code-ai-applications/translations/ja-jp/README.md b/10-building-low-code-ai-applications/translations/ja-jp/README.md index 07c31002b..ee32f3991 100644 --- a/10-building-low-code-ai-applications/translations/ja-jp/README.md +++ b/10-building-low-code-ai-applications/translations/ja-jp/README.md @@ -1,8 +1,8 @@ -# ローコードAIアプリケーションの構築 +# ローコード AI アプリケーションの構築 -[![Building Low Code AI Applications](../../images/10-lesson-banner.png?WT.mc_id=academic-105485-yoterada)](https://youtu.be/XX8491SAF44?WT.mc_id=academic-105485-yoterada) +[![Building Low Code AI Applications](../../images/10-lesson-banner.png?WT.mc_id=academic-105485-yoterada)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=170c6cb7-3c96-40f2-a354-b6f0ec523de7?WT.mc_id=academic-105485-yoterada) -> *(上記の画像をクリックすると、レッスン・ビデオを表示します)* +> _(上記の画像をクリックすると、レッスン・ビデオを表示します)_ ## はじめに @@ -47,13 +47,13 @@ Power Platform は生成 AI を用いて、ローコード開発とアプリケ ### どのように動作するのでしょうか? -Copilot は AI アシスタントで、自然言語を使って会話形式のやり取りで要件を説明し、Power Platform のソリューションを作れます。たとえば、アプリで使用するフィールドを AI アシスタントに指示すると、アプリとそれに基づくデータモデルの両方を作成します。また、Power Automateでフローを設定する方法を指定できます。 +Copilot は AI アシスタントで、自然言語を使って会話形式のやり取りで要件を説明し、Power Platform のソリューションを作れます。たとえば、アプリで使用するフィールドを AI アシスタントに指示すると、アプリとそれに基づくデータモデルの両方を作成します。また、Power Automate でフローを設定する方法を指定できます。 アプリ画面に Copilot の機能を組み込み、利用者は、会話型のやり取りを通じて、知見が得られるようになります。 AI Builder は、Power Platform で利用可能なローコード AI 機能で、AI モデルを使用してプロセスを自動化し、結果を予測できます。AI Builder を使用すると、Dataverse や SharePoint、OneDrive、Azure などの様々なクラウド上のデータソースに接続するアプリやフローに AI を組み込めます。 -Copilotは、Power Apps、Power Automate、Power BI、Power Pages、Power Virtual Agentsといった、全 Power Platform 製品で利用できます。一方で、AI Builder は Power Apps と Power Automate で利用できます。今回のレッスンでは、Power Apps と Power Automate における Copilot と AI Builder の使用方法について、特に教育スタートアップ用のソリューション開発の観点から詳しく説明していきます。 +Copilot は、Power Apps、Power Automate、Power BI、Power Pages、Power Virtual Agents といった、全 Power Platform 製品で利用できます。一方で、AI Builder は Power Apps と Power Automate で利用できます。今回のレッスンでは、Power Apps と Power Automate における Copilot と AI Builder の使用方法について、特に教育スタートアップ用のソリューション開発の観点から詳しく説明していきます。 ### Power Apps における Copilot @@ -83,25 +83,25 @@ Copilot を使い始める際には、[Power Platform Copilot Prompt Library](ht 1. ホーム画面のテキストエリアに、作成したいアプリを説明します。例えば、「**学生の課題を追跡し管理するアプリを作成してください**」と入力します。「**Send**」ボタンをクリックして、プロンプトを AI Copilot に送信します。 - ![Describe the app you want to build](../../images/copilot-chat-prompt-powerapps.png?WT.mc_id=academic-105485-yoterada) +![Describe the app you want to build](../../images/copilot-chat-prompt-powerapps.png?WT.mc_id=academic-105485-yoterada) 1. AI Copilot は、追跡したいデータを保存するために必要なフィールドとサンプルデータを持つ Dataverse テーブルの作成を提案します。その後、会話でのやり取りを通じ AI Copilot のアシスタント機能を使用して、テーブルを自分のニーズに合わせてカスタマイズできます。 - > **重要**:Dataverse は Power Platform の基盤となるデータプラットフォームで、アプリのデータを保存するためのローコード・データ・プラットフォームです。Microsoft Cloud 上にデータを安全に保存する、完全にマネージドなサービスで、Power Platform 環境内でプロビジョニングできます。データ分類、データ系統、きめ細やかなアクセス制御など、組み込みのデータガバナンス機能が付属しています。Dataverse については[こちら](https://docs.microsoft.com/powerapps/maker/data-platform/data-platform-intro?WT.mc_id=academic-109639-yoterada)から詳しく学ぶべます。 + > **重要**:Dataverse は Power Platform の基盤となるデータプラットフォームで、アプリのデータを保存するためのローコード・データ・プラットフォームです。Microsoft Cloud 上にデータを安全に保存する、完全にマネージドなサービスで、Power Platform 環境内でプロビジョニングできます。データ分類、データ系統、きめ細やかなアクセス制御など、組み込みのデータガバナンス機能が付属しています。Dataverse については[こちら](https://docs.microsoft.com/powerapps/maker/data-platform/data-platform-intro?WT.mc_id=academic-109639-yoterada)から詳しく学ぶべます。 ![Suggested fields in your new table](../../images/copilot-dataverse-table-powerapps.png?WT.mc_id=academic-105485-yoterada) -1. 教育者は、課題を提出した学生に進行状況を伝えるためにメールを送りたいと考えています。その為に、Copilotを使用して、学生のメールアドレスを保存する新しいフィールドをテーブルに追加できます。例えば、「**学生のメールを保存する列を追加してください**」というプロンプトを使用して、新しいフィールドを追加できます。「**Send**」ボタンをクリックして、プロンプトを AI Copilot に送信します。 +1. 教育者は、課題を提出した学生に進行状況を伝えるためにメールを送りたいと考えています。その為に、Copilot を使用して、学生のメールアドレスを保存する新しいフィールドをテーブルに追加できます。例えば、「**学生のメールを保存する列を追加してください**」というプロンプトを使用して、新しいフィールドを追加できます。「**Send**」ボタンをクリックして、プロンプトを AI Copilot に送信します。 ![Adding a new field](../../images/copilot-new-column.png?WT.mc_id=academic-105485-yoterada) -1. AI Copilot は新しいフィールドを生成し、その後、フィールドを自身のニーズに合わせてカスタマイズできます。 +1. AI Copilot は新しいフィールドを生成し、その後、フィールドを自身のニーズに合わせてカスタマイズできます。 1. テーブルが完成した後、「**Create app**」ボタンをクリックしてアプリを作成します。 1. AI Copilot は記述内容に基づいてレスポンシブな Canvas アプリを生成します。その後、アプリを自分のニーズに合わせてカスタマイズできます。 -1. 教育者が学生にメールを送るために、Copilotを使用してアプリに新しい画面を追加できます。例えば、次のプロンプトを使用してアプリに新しい画面を追加できます:「***学生にメールを送る画面を追加してください***」そして「**Send**」ボタンをクリックして、プロンプトを AI Copilot に送信します。 +1. 教育者が学生にメールを送るために、Copilot を使用してアプリに新しい画面を追加できます。例えば、次のプロンプトを使用してアプリに新しい画面を追加できます:「**_学生にメールを送る画面を追加してください_**」そして「**Send**」ボタンをクリックして、プロンプトを AI Copilot に送信します。 ![Adding a new screen via a prompt instruction](../../images/copilot-new-screen.png?WT.mc_id=academic-105485-yoterada) @@ -131,9 +131,9 @@ Power Platform には、アプリやソリューションのデータを保存 Dataverse とは何か、そしてその利点を理解したところで、Copilot を使って Dataverse でテーブルを作成し、財務チームの要件を満たすテーブル作成方法を見ていきましょう。 ->**注**:次のセクションでは、このテーブルを使って、すべての請求書情報を抽出し、そこからテーブルに保存するまでの自動化処理を構築します。 +> **注**:次のセクションでは、このテーブルを使って、すべての請求書情報を抽出し、そこからテーブルに保存するまでの自動化処理を構築します。 -Copilotを使って Dataverse でテーブルを作成する手順は下記の通りです: +Copilot を使って Dataverse でテーブルを作成する手順は下記の通りです: 1. [Power Apps](https://make.powerapps.com?WT.mc_id=academic-105485-yoterada) のホーム画面に遷移します。 @@ -190,7 +190,7 @@ AI Builder の請求書処理 AI モデルを使って、財務チームが請 2. ホーム画面のテキストエリアを使って、構築したいワークフローを記述します。例えば、「メールボックスに請求書が届いたら処理を開始してください」と入力します。「送信」ボタンをクリックして、この指示を AI Copilot に送ります。 - ![Copilot power automate](../../images/copilot-chat-prompt-powerautomate.png?WT.mc_id=academic-105485-yoterada) + ![Copilot power automate](../../images/copilot-chat-prompt-powerautomate.png?WT.mc_id=academic-105485-yoterada) 3. AI Copilot は、自動化したいタスクを実行するために必要なアクションを提案します。「次へ」ボタンをクリックすると、次のステップに進めます。 @@ -204,7 +204,7 @@ AI Builder の請求書処理 AI モデルを使って、財務チームが請 8. 「条件」アクションも使用しないため、フローから削除します。次のスクリーンショットのようになります。 - ![power automate, remove actions](../../images/powerautomate-remove-actions.png?WT.mc_id=academic-105485-yoterada) + ![power automate, remove actions](../../images/powerautomate-remove-actions.png?WT.mc_id=academic-105485-yoterada) 9. 「アクションの追加」ボタンをクリックし、「Dataverse」を検索します。「新しい行の追加」アクションを選択します。 diff --git a/10-building-low-code-ai-applications/translations/pt-br/README.md b/10-building-low-code-ai-applications/translations/pt-br/README.md index 9bff41291..00bf87a15 100644 --- a/10-building-low-code-ai-applications/translations/pt-br/README.md +++ b/10-building-low-code-ai-applications/translations/pt-br/README.md @@ -1,8 +1,8 @@ # Criando Aplication de IA com Low Code -[![Building Low Code AI Applications](../../images/10-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/XX8491SAF44?WT.mc_id=academic-105485-koreyst) +[![Building Low Code AI Applications](../../images/10-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=170c6cb7-3c96-40f2-a354-b6f0ec523de7?WT.mc_id=academic-105485-koreyst) -> *(Clique na imagem acima para ver o vídeo desta lição)* +> _(Clique na imagem acima para ver o vídeo desta lição)_ ## Introdução @@ -30,7 +30,7 @@ As ferramentas e tecnologias que você usará nesta lição são: ## IA Generativa na Power Platform -O aprimoramento do desenvolvimento de low-code e aplicativos com IA generativa é uma área-chave para a Power Platform. O objetivo é permitir que todos criam aplicativos, sites, painéis e automatizem processos com IA, *sem exigir qualquer conhecimento em ciência de dados*. Esse objetivo é alcançado integrando a IA generativa na experiência de desenvolvimento de low-code na Power Platform na forma de Copilot e AI Builder. +O aprimoramento do desenvolvimento de low-code e aplicativos com IA generativa é uma área-chave para a Power Platform. O objetivo é permitir que todos criam aplicativos, sites, painéis e automatizem processos com IA, _sem exigir qualquer conhecimento em ciência de dados_. Esse objetivo é alcançado integrando a IA generativa na experiência de desenvolvimento de low-code na Power Platform na forma de Copilot e AI Builder. ### Como isso funciona? @@ -68,17 +68,17 @@ Você criará o aplicativo usando Copilot no Power Apps seguindo as etapas abaix 1. Acesse a [tela inicial do Power Apps](https://make.powerapps.com?WT.mc_id=academic-105485-koreyst). -2. Use a área de texto na tela inicial para descrever o aplicativo que deseja criar. Por exemplo, ***Quero criar um aplicativo para rastrear e gerenciar as atribuições de estudantes***. Clique no botão **Enviar** para enviar o prompt ao AI Copilot. +2. Use a área de texto na tela inicial para descrever o aplicativo que deseja criar. Por exemplo, **_Quero criar um aplicativo para rastrear e gerenciar as atribuições de estudantes_**. Clique no botão **Enviar** para enviar o prompt ao AI Copilot. ![Descreva o aplicativo que deseja criar](../../images/copilot-chat-prompt-powerapps.png?WT.mc_id=academic-105485-koreyst) 1. The AI Copilot will suggest a Dataverse Table with the fields you need to store the data you want to track and some sample data. You can then customize the table to meet your needs using the AI Copilot assistant feature through conversational steps. - > **Important**: Dataverse is the underlying data platform for Power Platform. It is a low-code data platform for storing the app's data. It is a fully managed service that securely stores data in the Microsoft Cloud and is provisioned within your Power Platform environment. It comes with built-in data governance capabilities, such as data classification, data lineage, fine-grained access control, and more. You can learn more about Dataverse [here](https://docs.microsoft.com/powerapps/maker/data-platform/data-platform-intro?WT.mc_id=academic-109639-somelezediko). + > **Important**: Dataverse is the underlying data platform for Power Platform. It is a low-code data platform for storing the app's data. It is a fully managed service that securely stores data in the Microsoft Cloud and is provisioned within your Power Platform environment. It comes with built-in data governance capabilities, such as data classification, data lineage, fine-grained access control, and more. You can learn more about Dataverse [here](https://docs.microsoft.com/powerapps/maker/data-platform/data-platform-intro?WT.mc_id=academic-109639-somelezediko). ![Suggested fields in your new table](../../images/copilot-dataverse-table-powerapps.png?WT.mc_id=academic-105485-koreyst) -1. Os educadores desejam enviar e-mails aos alunos que enviaram suas atribuições para mantê-los atualizados sobre o progresso de suas tarefas. Você pode usar o Copilot para adicionar um novo campo à tabela para armazenar o e-mail do aluno. Por exemplo, você pode usar o seguinte comando para adicionar um novo campo à tabela: ***Quero adicionar uma coluna para armazenar o e-mail do aluno***. Clique no botão **Enviar** para enviar o comando ao AI Copilot. +1. Os educadores desejam enviar e-mails aos alunos que enviaram suas atribuições para mantê-los atualizados sobre o progresso de suas tarefas. Você pode usar o Copilot para adicionar um novo campo à tabela para armazenar o e-mail do aluno. Por exemplo, você pode usar o seguinte comando para adicionar um novo campo à tabela: **_Quero adicionar uma coluna para armazenar o e-mail do aluno_**. Clique no botão **Enviar** para enviar o comando ao AI Copilot. ![Adicionando um novo campo](../../images/copilot-new-column.png?WT.mc_id=academic-105485-koreyst) @@ -88,7 +88,7 @@ Você criará o aplicativo usando Copilot no Power Apps seguindo as etapas abaix 1. O AI Copilot gerará um aplicativo Canvas responsivo com base na sua descrição. Você pode, então, personalizar o aplicativo conforme suas necessidades. -1. Para que os educadores possam enviar e-mails aos alunos, você pode usar o Copilot para adicionar uma nova tela ao aplicativo. Por exemplo, você pode usar o seguinte comando para adicionar uma nova tela ao aplicativo: ***Quero adicionar uma tela para enviar e-mails aos alunos***. Clique no botão **Enviar** para enviar o comando ao AI Copilot. +1. Para que os educadores possam enviar e-mails aos alunos, você pode usar o Copilot para adicionar uma nova tela ao aplicativo. Por exemplo, você pode usar o seguinte comando para adicionar uma nova tela ao aplicativo: **_Quero adicionar uma tela para enviar e-mails aos alunos_**. Clique no botão **Enviar** para enviar o comando ao AI Copilot. ![Adicionando uma nova tela via comando](../../images/copilot-new-screen.png?WT.mc_id=academic-105485-koreyst) @@ -118,7 +118,7 @@ Por que devemos usar o Dataverse para nossa startup? As tabelas padrão e person Agora que você sabe o que é o Dataverse e por que deve usá-lo, vamos ver como você pode usar o Copilot para criar uma tabela no Dataverse para atender aos requisitos de nossa equipe financeira. ->**Nota** : Você usará esta tabela na próxima seção para criar uma automação que extrairá todas as informações da fatura e as armazenará na tabela. +> **Nota** : Você usará esta tabela na próxima seção para criar uma automação que extrairá todas as informações da fatura e as armazenará na tabela. Para criar uma tabela no Dataverse usando o Copilot, siga as etapas abaixo: @@ -128,7 +128,7 @@ Para criar uma tabela no Dataverse usando o Copilot, siga as etapas abaixo: ![Selecionar nova tabela](../../images/describe-new-table.png?WT.mc_id=academic-105485-koreyst) -1. Na tela **Descrever a nova tabela**, use a área de texto para descrever a tabela que deseja criar. Por exemplo, ***Quero criar uma tabela para armazenar informações de fatura***. Clique no botão **Enviar** para enviar o comando ao AI Copilot. +1. Na tela **Descrever a nova tabela**, use a área de texto para descrever a tabela que deseja criar. Por exemplo, **_Quero criar uma tabela para armazenar informações de fatura_**. Clique no botão **Enviar** para enviar o comando ao AI Copilot. ![Descrever a tabela](../../images/copilot-chat-prompt-dataverse.png?WT.mc_id=academic-105485-koreyst) @@ -136,7 +136,7 @@ Para criar uma tabela no Dataverse usando o Copilot, siga as etapas abaixo: ![Tabela Dataverse sugerida](../../images/copilot-dataverse-table.png?WT.mc_id=academic-105485-koreyst) -1. A equipe financeira deseja enviar um e-mail ao fornecedor para atualizá-lo com o status atual de sua fatura. Você pode usar o Copilot para adicionar um novo campo à tabela para armazenar o e-mail do fornecedor. Por exemplo, você pode usar o seguinte comando para adicionar um novo campo à tabela: ***Quero adicionar uma coluna para armazenar o e-mail do fornecedor***. Clique no botão **Enviar** para enviar o comando ao AI Copilot. +1. A equipe financeira deseja enviar um e-mail ao fornecedor para atualizá-lo com o status atual de sua fatura. Você pode usar o Copilot para adicionar um novo campo à tabela para armazenar o e-mail do fornecedor. Por exemplo, você pode usar o seguinte comando para adicionar um novo campo à tabela: **_Quero adicionar uma coluna para armazenar o e-mail do fornecedor_**. Clique no botão **Enviar** para enviar o comando ao AI Copilot. 1. O AI Copilot gerará um novo campo e você poderá personalizá-lo conforme suas necessidades. @@ -175,9 +175,9 @@ Para criar um fluxo que ajudará a equipe financeira a processar faturas usando 1. Acesse a [tela inicial do Power Automate](https://make.powerautomate.com?WT.mc_id=academic-105485-koreyst). -2. Use a área de texto na tela inicial para descrever o fluxo que você deseja criar. Por exemplo, ***Processar uma fatura quando ela chegar à minha caixa de correio***. Clique no botão **Enviar** para enviar o comando ao AI Copilot. +2. Use a área de texto na tela inicial para descrever o fluxo que você deseja criar. Por exemplo, **_Processar uma fatura quando ela chegar à minha caixa de correio_**. Clique no botão **Enviar** para enviar o comando ao AI Copilot. - ![Copilot power automate](../../images/copilot-chat-prompt-powerautomate.png?WT.mc_id=academic-105485-koreyst) + ![Copilot power automate](../../images/copilot-chat-prompt-powerautomate.png?WT.mc_id=academic-105485-koreyst) 3. O AI Copilot sugerirá as ações que você precisa realizar para a tarefa que deseja automatizar. Você pode clicar no botão **Próximo** para passar para as próximas etapas. @@ -191,7 +191,7 @@ Para criar um fluxo que ajudará a equipe financeira a processar faturas usando 8. Remova a ação **Condição** do fluxo, pois você não a usará. Deve se parecer com a captura de tela a seguir: - ![Power Automate, remover ações](../../images/powerautomate-remove-actions.png?WT.mc_id=academic-105485-koreyst) + ![Power Automate, remover ações](../../images/powerautomate-remove-actions.png?WT.mc_id=academic-105485-koreyst) 9. Clique no botão **Adicionar uma ação** e pesquise por **Dataverse**. Selecione a ação **Adicionar uma nova linha**. diff --git a/12-designing-ux-for-ai-applications/README.md b/12-designing-ux-for-ai-applications/README.md index b7215cb24..23f16e286 100644 --- a/12-designing-ux-for-ai-applications/README.md +++ b/12-designing-ux-for-ai-applications/README.md @@ -1,10 +1,10 @@ # Designing UX for AI Applications -[![Designing UX for AI Applications](./images/12-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/bO7h2_hOhR0?WT.mc_id=academic-105485-koreyst) +[![Designing UX for AI Applications](./images/12-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=1464ef7a-d974-4a04-9ce3-1bc5b3087b4b?WT.mc_id=academic-105485-koreyst) -> *(Click the image above to view video of this lesson)* +> _(Click the image above to view video of this lesson)_ -User experience is a very important aspect of building apps. Users need to be able to use your app in an efficient way to perform tasks. Being efficient is one thing but you also need to design apps so that they can be used by everyone, to make them *accessible*. This chapter will focus on this area so you hopefully end up designing an app that people can and want to use. +User experience is a very important aspect of building apps. Users need to be able to use your app in an efficient way to perform tasks. Being efficient is one thing but you also need to design apps so that they can be used by everyone, to make them _accessible_. This chapter will focus on this area so you hopefully end up designing an app that people can and want to use. ## Introduction @@ -12,16 +12,16 @@ User experience is how a user interacts with and uses a specific product or serv The lesson will cover the following areas: -* Introduction to User Experience and Understanding User Needs -* Designing AI Applications for Trust and Transparency -* Designing AI Applications for Collaboration and Feedback +- Introduction to User Experience and Understanding User Needs +- Designing AI Applications for Trust and Transparency +- Designing AI Applications for Collaboration and Feedback ## Learning goals After taking this lesson, you'll be able to: -* Understand how to build AI applications that meet the user needs. -* Design AI applications that promote trust and collaboration. +- Understand how to build AI applications that meet the user needs. +- Design AI applications that promote trust and collaboration. ### Prerequisite @@ -103,13 +103,13 @@ AI applications are not perfect, therefore, they are bound to make mistakes. Whe Take any AI apps you've built so far, consider implementing the below steps in your app: -* **Pleasant:** Consider how you can make your app more pleasant. Are you adding explanations everywhere, are you encouraging the user to explore? How are you wording your error messages? +- **Pleasant:** Consider how you can make your app more pleasant. Are you adding explanations everywhere, are you encouraging the user to explore? How are you wording your error messages? -* **Usability:** Building a web app. Make sure your app is navigable by both mouse and keyboard. +- **Usability:** Building a web app. Make sure your app is navigable by both mouse and keyboard. -* **Trust and transparency:** Don't trust the AI completely and its output, consider how you would add a human to the process to verify the output. Also, consider and implement other ways to achieve trust and transparency. +- **Trust and transparency:** Don't trust the AI completely and its output, consider how you would add a human to the process to verify the output. Also, consider and implement other ways to achieve trust and transparency. -* **Control:** Give the user control of the data they provide to the application. Implement a way a user can opt-in and opt-out of data collection in the AI application. +- **Control:** Give the user control of the data they provide to the application. Implement a way a user can opt-in and opt-out of data collection in the AI application. @@ -117,4 +117,4 @@ Take any AI apps you've built so far, consider implementing the below steps in y After completing this lesson, check out our [Generative AI Learning collection](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) to continue leveling up your Generative AI knowledge! -Congratulations, you have completed this course! The building should not stop here. Hopefully, you have been inspired to start building your own Generative AI startup. Head over to the [Microsoft Founders Hub](https://aka.ms/genai-foundershub?WT.mc_id=academic-105485-koreyst) and apply for the program to receive support on your journey. \ No newline at end of file +Congratulations, you have completed this course! The building should not stop here. Hopefully, you have been inspired to start building your own Generative AI startup. Head over to the [Microsoft Founders Hub](https://aka.ms/genai-foundershub?WT.mc_id=academic-105485-koreyst) and apply for the program to receive support on your journey. diff --git a/12-designing-ux-for-ai-applications/translations/cn/README.md b/12-designing-ux-for-ai-applications/translations/cn/README.md index 0b056b2f6..a184f1608 100644 --- a/12-designing-ux-for-ai-applications/translations/cn/README.md +++ b/12-designing-ux-for-ai-applications/translations/cn/README.md @@ -1,29 +1,27 @@ # 第十二章:为人工智能应用程序添加用户体验 -[![Designing UX for AI Applications](../../images/12-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/bO7h2_hOhR0?WT.mc_id=academic-105485-koreyst) +[![Designing UX for AI Applications](../../images/12-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=1464ef7a-d974-4a04-9ce3-1bc5b3087b4b?WT.mc_id=academic-105485-koreyst) - -*(点击该图片看本章导学视频)* +_(点击该图片看本章导学视频)_ 用户体验是构建应用程序的一个非常重要的方面。 用户需要能够有效地使用您的应用程序来执行任务。 高效是一回事,但您还需要设计应用程序,以便每个人都可以使用它们,使它们“易于访问”。 本章将重点讨论这一领域,希望最终设计出一个人们可以并且想要使用的应用程序。 - ## 本章概述 用户体验是指用户如何与特定产品或服务(无论是系统、工具还是设计)进行交互和使用。 在开发人工智能应用程序时,开发人员不仅注重确保用户体验有效,而且注重道德。 在本课程中,我们将介绍如何构建满足用户需求的人工智能 (AI) 应用程序。 本课程将包含以下内容: -* 用户体验简介和了解用户需求 -* 设计人工智能应用程序以实现信任和透明度 -* 设计用于协作和反馈的人工智能应用程序 +- 用户体验简介和了解用户需求 +- 设计人工智能应用程序以实现信任和透明度 +- 设计用于协作和反馈的人工智能应用程序 ## 学习目标 学习完本课程后,您将会了解到: -* 了解如何构建满足用户需求的人工智能应用程序。 -* 设计促进信任和协作的人工智能应用程序。 +- 了解如何构建满足用户需求的人工智能应用程序。 +- 设计促进信任和协作的人工智能应用程序。 ### 先决条件 @@ -87,7 +85,6 @@ Bing 中允许用户控制应用程序的另一个功能是能够选择加入和 > 在设计人工智能应用程序时,意向性是确保用户不会过度信任对其功能设定不切实际的期望的关键。 做到这一点的一种方法是在提示和结果之间制造摩擦。 提醒用户这是人工智能而不是人类 - ## 设计用于协作和反馈的 AI 应用程序 正如前面提到的,生成式人工智能在用户和人工智能之间建立了协作。 大多数交互都是由用户输入提示,然后人工智能生成输出。 如果输出不正确怎么办? 如果发生错误,应用程序如何处理? 人工智能会责怪用户还是需要时间来解释错误? @@ -106,17 +103,16 @@ Bing 中允许用户控制应用程序的另一个功能是能够选择加入和 对于您迄今为止构建的任何 AI 应用程序,请考虑在您的应用程序中实施以下步骤: -* **满意度:** 考虑如何使您的应用程序更加令人愉快。 您是否到处添加解释,是否鼓励用户探索? 您如何措辞错误消息? - -* **可用性:** 构建网络应用程序。 确保您的应用程序可以通过鼠标和键盘进行导航。 +- **满意度:** 考虑如何使您的应用程序更加令人愉快。 您是否到处添加解释,是否鼓励用户探索? 您如何措辞错误消息? -* **信任和透明度:** 不要完全信任人工智能及其输出,考虑如何在流程中添加人员来验证输出。 此外,考虑并实施其他方法来实现信任和透明度。 +- **可用性:** 构建网络应用程序。 确保您的应用程序可以通过鼠标和键盘进行导航。 -* **可控:** 赋予用户对其提供给应用程序的数据的控制权。 实现用户可以选择加入和退出 AI 应用程序中的数据收集的方式。 +- **信任和透明度:** 不要完全信任人工智能及其输出,考虑如何在流程中添加人员来验证输出。 此外,考虑并实施其他方法来实现信任和透明度。 +- **可控:** 赋予用户对其提供给应用程序的数据的控制权。 实现用户可以选择加入和退出 AI 应用程序中的数据收集的方式。 ## 恭喜您完成本课程 想要了解更多有关为人工智能应用程序设计用户体验的信息吗? 转至[继进阶学习的页面续学习页面](../../../13-continued-learning/translations/cn/README.md?WT.mc_id=academic-105485-koreyst) 查找有关此主题的其他重要资源。 -恭喜,您已完成本课程! 你不应该学完即止。 希望您受到启发,开始创建自己的生成式人工智能初创公司。 前往 [Microsoft Founders Hub](https://aka.ms/genai-Foundershub?WT.mc_id=academic-105485-koreyst) 并申请该计划以并获得支持。 \ No newline at end of file +恭喜,您已完成本课程! 你不应该学完即止。 希望您受到启发,开始创建自己的生成式人工智能初创公司。 前往 [Microsoft Founders Hub](https://aka.ms/genai-Foundershub?WT.mc_id=academic-105485-koreyst) 并申请该计划以并获得支持。 diff --git a/12-designing-ux-for-ai-applications/translations/ja-jp/README.md b/12-designing-ux-for-ai-applications/translations/ja-jp/README.md index 7bf019a57..07dc7e717 100644 --- a/12-designing-ux-for-ai-applications/translations/ja-jp/README.md +++ b/12-designing-ux-for-ai-applications/translations/ja-jp/README.md @@ -1,8 +1,8 @@ # AI アプリケーションの UX 設計 -[![Designing UX for AI Applications](../../images/12-lesson-banner.png?WT.mc_id=academic-105485-yoterada)](https://youtu.be/bO7h2_hOhR0?WT.mc_id=academic-105485-yoterada) +[![Designing UX for AI Applications](../../images/12-lesson-banner.png?WT.mc_id=academic-105485-yoterada)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=1464ef7a-d974-4a04-9ce3-1bc5b3087b4b?WT.mc_id=academic-105485-yoterada) -> *(上の画像をクリックすると、このレッスンのビデオが表示されます)* +> _(上の画像をクリックすると、このレッスンのビデオが表示されます)_ アプリを構築する際、ユーザー・エクスペリエンスは非常に重要な要素です。利用者は、タスクを効率的に実行できるようにアプリを使用する必要があります。効率性は一つの要素ですが、全ての人が使用でき、アクセスできるようにアプリを設計するのも重要です。この章では、この領域に焦点を当て、すべての利用者が使用でき、使いたくなるアプリの設計を目指します。 @@ -12,16 +12,16 @@ このレッスンでは、下記の内容について説明します。 -* ユーザー・エクスペリエンスの概要と利用者のニーズの理解 -* 信頼性と透明性のある AI アプリケーションの設計 -* コラボレーションとフィードバックを求める AI アプリケーションの設計 +- ユーザー・エクスペリエンスの概要と利用者のニーズの理解 +- 信頼性と透明性のある AI アプリケーションの設計 +- コラボレーションとフィードバックを求める AI アプリケーションの設計 ## 学習目標 このレッスンを修了すると、下記を理解できます: -* 利用者のニーズを満たす AI アプリケーションを構築する方法を理解します。 -* 信頼性とコラボレーションを促進する AI アプリケーションを設計します。 +- 利用者のニーズを満たす AI アプリケーションを構築する方法を理解します。 +- 信頼性とコラボレーションを促進する AI アプリケーションを設計します。 ### 前提 @@ -39,7 +39,7 @@ ### 信頼性 -信頼性が高いとは、アプリケーションが一貫してエラーなくタスクを実行できる状況をさします。しかし、AIも人間と同じく完全ではなく、エラーが起こる可能性があります。アプリケーションは、人間の介入や修正が必要なエラーや、予期せぬ状況に遭遇する場合もあります。では、エラーはどのように取り扱うべきでしょうか?このレッスンの最後のセクションでは、AI システムとアプリケーションが、どのようにして協力やフィードバックを設計に取り入れているかについて説明します。 +信頼性が高いとは、アプリケーションが一貫してエラーなくタスクを実行できる状況をさします。しかし、AI も人間と同じく完全ではなく、エラーが起こる可能性があります。アプリケーションは、人間の介入や修正が必要なエラーや、予期せぬ状況に遭遇する場合もあります。では、エラーはどのように取り扱うべきでしょうか?このレッスンの最後のセクションでは、AI システムとアプリケーションが、どのようにして協力やフィードバックを設計に取り入れているかについて説明します。 ### アクセシビリティ @@ -53,7 +53,7 @@ 全ての問題を AI で解決できるわけではありません。AI は、手動の作業を自動化したり、利用者体験をパーソナライズし、利用者体験を向上させる役割を果たします。 -## 信頼性と透明性のためのAIアプリケーションの設計 +## 信頼性と透明性のための AI アプリケーションの設計 AI アプリケーションを設計する際には、利用者との信頼関係の構築が重要です。信頼があると、利用者はアプリケーションがタスクを遂行し、一貫した結果を提供し、その結果が自分のニーズに合致すると確信できます。この分野でのリスクは、不信と過信です。不信感は、利用者が AI システムをほとんど信頼していない、または全く信頼していない場合に生じ、結果として利用者がアプリケーションの利用を拒否するのにつながります。一方、過信は、利用者が AI システムの能力を過大評価し、AI システムを過度に信頼する状況を指します。例えば、過信があると、自動採点システムを使う教師は、採点システムが適切に機能しているか確認するために、一部の試験を見直さない可能性があります。これにより、学生に対する不公平な評価や誤った評価が生じる可能性があるため、フィードバックや改善の機会を逃す可能性があります。 @@ -105,13 +105,13 @@ AI アプリケーションは完全ではなく、間違いを犯すのは避 これまでに構築した AI アプリを使用して、アプリに次の実装を加えてください。 -* **快適:** あなたのアプリをより使いやすくするためにどのような工夫ができるか考えてみてください。各所に説明を追加していますか?利用者が自由に探索できるように促していますか?エラーメッセージはどのように表現していますか? +- **快適:** あなたのアプリをより使いやすくするためにどのような工夫ができるか考えてみてください。各所に説明を追加していますか?利用者が自由に探索できるように促していますか?エラーメッセージはどのように表現していますか? -* **ユーザビリティ:** Web アプリ構築の場合、アプリがマウスとキーボードの両方で操作できていますか? +- **ユーザビリティ:** Web アプリ構築の場合、アプリがマウスとキーボードの両方で操作できていますか? -* **信頼と透明性:** AI とその結果を完全に信頼するのではなく、結果を確認するために人間が介入するプロセスをどのように追加するか考えてみてください。また、信頼性と透明性を高めるための他の手段も検討し、取り入れてみてください。 +- **信頼と透明性:** AI とその結果を完全に信頼するのではなく、結果を確認するために人間が介入するプロセスをどのように追加するか考えてみてください。また、信頼性と透明性を高めるための他の手段も検討し、取り入れてみてください。 -* **コントロール:** 利用者が自分で提供するデータを管理できるようにしましょう。AI アプリケーションで、利用者が使用するデータの選択(オプトイン)や選択解除(オプトアウト)する方法を導入してください。 +- **コントロール:** 利用者が自分で提供するデータを管理できるようにしましょう。AI アプリケーションで、利用者が使用するデータの選択(オプトイン)や選択解除(オプトアウト)する方法を導入してください。 diff --git a/12-designing-ux-for-ai-applications/translations/pt-br/README.md b/12-designing-ux-for-ai-applications/translations/pt-br/README.md index ebe8f450f..70d949c6a 100644 --- a/12-designing-ux-for-ai-applications/translations/pt-br/README.md +++ b/12-designing-ux-for-ai-applications/translations/pt-br/README.md @@ -1,10 +1,10 @@ # Projetando UX para aplicativos de IA -[![Designing UX for AI Applications](../../images/12-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://youtu.be/bO7h2_hOhR0?WT.mc_id=academic-105485-koreyst) +[![Designing UX for AI Applications](../../images/12-lesson-banner.png?WT.mc_id=academic-105485-koreyst)](https://learn.microsoft.com/_themes/docs.theme/master/en-us/_themes/global/video-embed.html?id=1464ef7a-d974-4a04-9ce3-1bc5b3087b4b?WT.mc_id=academic-105485-koreyst) -> *(Clique na imagem acima para assistir ao vídeo desta lição)* +> _(Clique na imagem acima para assistir ao vídeo desta lição)_ -A experiência do usuário é um aspecto muito importante na criação de aplicativos. Os usuários precisam poder usar seu aplicativo de maneira eficiente para realizar tarefas. Ser eficiente é uma coisa, mas você também precisa projetar aplicativos para que possam ser usados por todos, tornando-os *acessíveis*. Este capítulo se concentrará nessa área para que você termine projetando um aplicativo que as pessoas possam e queiram usar. +A experiência do usuário é um aspecto muito importante na criação de aplicativos. Os usuários precisam poder usar seu aplicativo de maneira eficiente para realizar tarefas. Ser eficiente é uma coisa, mas você também precisa projetar aplicativos para que possam ser usados por todos, tornando-os _acessíveis_. Este capítulo se concentrará nessa área para que você termine projetando um aplicativo que as pessoas possam e queiram usar. ## Introdução @@ -12,16 +12,16 @@ A experiência do usuário é como um usuário interage e utiliza um produto ou A lição abordará as seguintes áreas: -* Introdução à Experiência do Usuário e Compreensão das Necessidades do Usuário -* Projeto de Aplicações de IA para Confiança e Transparência -* Projeto de Aplicações de IA para Colaboração e Feedback +- Introdução à Experiência do Usuário e Compreensão das Necessidades do Usuário +- Projeto de Aplicações de IA para Confiança e Transparência +- Projeto de Aplicações de IA para Colaboração e Feedback ## Metas de Aprendizado Após esta lição, você será capaz de: -* Compreender como construir aplicativos de IA que atendam às necessidades do usuário. -* Projetar aplicativos de IA que promovam confiança e colaboração. +- Compreender como construir aplicativos de IA que atendam às necessidades do usuário. +- Projetar aplicativos de IA que promovam confiança e colaboração. ### Pré-requisito diff --git a/13-securing-ai-applications/README.md b/13-securing-ai-applications/README.md new file mode 100644 index 000000000..22922dab5 --- /dev/null +++ b/13-securing-ai-applications/README.md @@ -0,0 +1,164 @@ +# Securing Your Generative AI Applications + +[![Securing Your Generative AI Applications](./images/13-lesson-banner.jpg?WT.mc_id=academic-105485-koreyst)]() + +## Introduction + +This lesson will cover: + +- Security within the context of AI systems. +- Common risks and threats to AI systems. +- Methods and considerations for securing AI systems. + +## Learning Goals + +After completing this lesson, you will have an understanding of: + +- The threats and risks to AI systems. +- Common methods and practices for securing AI systems. +- How implementing security testing can prevent unexpected results and erosion of user trust. + +## What does security mean within the context of generative AI? + +As Artificial Intelligence (AI) and Machine Learning (ML) technologies increasingly shape our lives, it’s crucial to protect not only customer data but also the AI systems themselves. AI/ML is increasingly used in support of high-value decision-making processes in industries where the wrong decision may result in serious consequences. + +Here are key points to consider: + +- **Impact of AI/ML**: AI and ML have significant impacts on daily life and as such safeguarding them has become essential. +- **Security Challenges**: This impact that AI/ML has needs proper attention in order to address the need to protect AI-based products from sophisticated attacks, whether by trolls or organized groups. +- **Strategic Problems**: The tech industry must proactively address strategic challenges to ensure long-term customer safety and data security. + +Additionally, Machine Learning models are largely unable to discern between malicious input and benign anomalous data. A significant source of training data is derived from uncurated, unmoderated, public datasets, which are open to 3rd-party contributions. Attackers don’t need to compromise datasets when they're free to contribute to them. Over time, low-confidence malicious data becomes high-confidence trusted data, if the data structure/formatting remains correct. + +This is why it is critical to ensure the integrity and protectiom of the data stores your models use to make decisions with. + +## Understanding the threats and risks of AI + +In terms of AI and related systems, data poisoning stands out as the most significant security threat today. This is due to the absence of standardized detection and mitigation methods, coupled with our reliance on untrusted or uncurated public datasets for training. To maintain data integrity and prevent a flawed training process, it is crucial to track the origin and lineage of your data. Otherwise, the old adage “garbage in, garbage out” holds true, leading to compromised model performance. + +Here are examples of how data poisoning can affect your models: + +1. **Label Flipping**: In a binary classification task, an adversary intentionally flips the labels of a small subset of training data. For instance, benign samples are labeled as malicious, leading the model to learn incorrect associations.\ + **Example**: A spam filter misclassifying legitimate emails as spam due to manipulated labels. +2. **Feature Poisoning**: An attacker subtly modifies features in the training data to introduce bias or mislead the model.\ + **Example**: Adding irrelevant keywords to product descriptions to manipulate recommendation systems. +3. **Data Injection**: Injecting malicious data into the training set to influence the model’s behavior.\ + **Example**: Introducing fake user reviews to skew sentiment analysis results. +4. **Backdoor Attacks**: An adversary inserts a hidden pattern (backdoor) into the training data. The model learns to recognize this pattern and behaves maliciously when triggered.\ + **Example**: A face recognition system trained with backdoored images that misidentifies a specific person. + +The MITRE Corporation has created [ATLAS (Adversarial Threat Landscape for Artificial-Intelligence Systems)](https://atlas.mitre.org/), a knowledgebase of tactics and techniques employed by adversaries in real-world attacks on AI systems. + +> There are a growing number of vulnerabilities in AI-enabled systems, as the incorporation of AI increases the attack surface of existing systems beyond those of traditional cyber-attacks. We developed ATLAS to raise awareness of these unique and evolving vulnerabilities, as the global community starts to incorporate AI into more systems. ATLAS is modeled after the MITRE ATT&CK® framework and its tactics, techniques, and procedures (TTPs) are complementary to those in ATT&CK. + +Much like the MITRE ATT&CK® framework, which is extensively used in traditional cybersecurity for planning advanced threat emulation scenarios, ATLAS provides an easily searchable set TTPs that can help to better understand and prepare for defending against emerging attacks. + +Additionally, the Open Web Application Security Project (OWASP) has created a "[Top 10 list](https://llmtop10.com/)" of the most critical vulnerabilities found in applications utilizing LLMs. The list highlights the risks of threats such as the aforementioned data poisoning along with others such as: + +- **Prompt Injection**: a technique where attackers manipulate a Large Language Model (LLM) through carefully crafted inputs, causing it to behave outside of its intended behavior. +- **Supply Chain Vulnerabilities**: The components and software that make up the applications used by an LLM, such as Python modules or external datasets, can themselves be compromised leading to unpexected results, introduced biases and even vulmerabilities in the underlying infrastructure. +- **Overreliance**: LLMs are fallible and have been prone to hallucinate, providing inaccurate or unsafe results. In several documented circumstances, people have taken the results at face value leading to unintended real-world megative consequences. + +Microsoft Cloud Advocate Rod Trent has written a free ebook, [Must Learn AI Security](https://github.com/rod-trent/OpenAISecurity/tree/main/Must_Learn/Book_Version), that dives deeply into these and other emerging AI threats and provides extensive guidance on how to best tackle these scenarios. + +## Security Testing for AI Systems and LLMs + +Artificial intelligence (AI) is transforming various domains and industries, offering new possibilities and benefits for society. However, AI also poses significant challenges and risks, such as data privacy, bias, lack of explainability, and potential misuse. Therefore, it is crucial to ensure that AI systems are secure and responsible, meaning that they adhere to ethical and legal standards and can be trusted by users and stakeholders. + +Security testing is the process of evaluating the security of an AI system or LLM by identifying and exploiting its vulnerabilities. This can be performed by developers, users, or third-party auditors, depending on the purpose and scope of the testing. Some of the most common security testing methods for AI systems and LLMs are: + +- **Data sanitization**: This is the process of removing or anonymizing sensitive or private information from the training data or the input of an AI system or LLM. Data sanitization can help prevent data leakage and malicious manipulation by reducing the exposure of confidential or personal data. +- **Adversarial testing**: This is the process of generating and applying adversarial examples to the input or output of an AI system or LLM to evaluate its robustness and resilience against adversarial attacks. Adversarial testing can help identify and mitigate the vulnerabilities and weaknesses of an AI system or LLM that may be exploited by attackers. +- **Model verification**: This is the process of verifying the correctness and completeness of the model parameters or architecture of an AI system or LLM. Model verification can help detect and prevent model stealing by ensuring that the model is protected and authenticated. +- **Output validation**: This is the process of validating the quality and reliability of the output of an AI system or LLM. Output validation can help detect and correct malicious manipulation by ensuring that the output is consistent and accurate. + +OpenAI, a leader in AI systems, has setup a series of _safety evaluations_ as part of their red teaming network initiative, aimed at testing the output AI systems in the hopes of contributing to AI safety. + +> Evaluations can range from simple Q&A tests to more-complex simulations. As concrete examples, here are sample evaluations developed by OpenAI for evaluating AI behaviors from a number of angles: + +#### Persuasion + +- [MakeMeSay](https://github.com/openai/evals/tree/main/evals/elsuite/make_me_say/readme.md): How well can an AI system trick another AI system into saying a secret word? +- [MakeMePay](https://github.com/openai/evals/tree/main/evals/elsuite/make_me_pay/readme.md): How well can an AI system convince another AI system to donate money? +- [Ballot Proposal](https://github.com/openai/evals/tree/main/evals/elsuite/ballots/readme.md): How well can an AI system influence another AI system’s support of a political proposition? + +#### Steganography (hidden messaging) + +- [Steganography](https://github.com/openai/evals/tree/main/evals/elsuite/steganography/readme.md): How well can an AI system ​​pass secret messages without being caught by another AI system? +- [Text Compression](https://github.com/openai/evals/tree/main/evals/elsuite/text_compression/readme.md): How well can an AI system compress and decompress messages, to enable hiding secret messages? +- [Schelling Point](https://github.com/openai/evals/blob/main/evals/elsuite/schelling_point/README.md): How well can an AI system coordinate with another AI system, without direct communication? + +### AI Security + +It's imperative that we aim to protect AI systems from malicious attacks, misuse, or unintended consequences. This includes taking steps to ensure the safety, reliability, and trustworthiness of AI systems, such as: + +- Securing the data and algorithms that are used to train and run AI models +- Preventing unauthorized access, manipulation, or sabotage of AI systems +- Detecting and mitigating bias, discrimination, or ethical issues in AI systems +- Ensuring the accountability, transparency, and explainability of AI decisions and actions +- Aligning the goals and values of AI systems with those of humans and society + +AI security is important for ensuring the integrity, availability, and confidentiality of AI systems and data. Some of the challenges and opportunities of AI security are: + +- Opportunity: Incorporating AI in cybersecurity strategies since it can play a crucial role in identifying threats and improving response times. AI can help automate and augment the detection and mitigation of cyberattacks, such as phishing, malware, or ransomware. +- Challenge: AI can also be used by adversaries to launch sophisticated attacks, such as generating fake or misleading content, impersonating users, or exploiting vulnerabilities in AI systems. Therefore, AI developers have a unique responsibility to design systems that are robust and resilient against misuse. + +### Data Protection + +LLMs can pose risks to the privacy and security of the data that they use. For example, LLMs can potentially memorize and leak sensitive information from their training data, such as personal names, addresses, passwords, or credit card numbers. They can also be manipulated or attacked by malicious actors who want to exploit their vulnerabilities or biases. Therefore, it is important to be aware of these risks and take appropriate measures to protect the data used with LLMs. There are several steps that you can take to protect the data that is used with LLMs. These steps include: + +- **Limiting the amount and type of data that they share with LLMs**: Only share the data that is necessary and relevant for the intended purposes, and avoid sharing any data that is sensitive, confidential, or personal. Users should also anonymize or encrypt the data that they share with LLMs, such as by removing or masking any identifying information, or using secure communication channels. +- **Verifying the data that LLMs generate**: Always check the accuracy and quality of the output generated by LLMs to ensure they don't contain any unwanted or inappropriate information. +- **Reporting and alerting any data breaches or incidents**: Be vigilant of any suspicious or abnormal activities or behaviors from LLMs, such as generating texts that are irrelevant, inaccurate, offensive, or harmful. This could be an indication of a data breach or security incident. + +Data security, governance, and compliance are critical for any organization that wants to leverage the power of data and AI in a multi-cloud environment. Securing and governing all your data is a complex and multifaceted undertaking. You need to secure and govern different types of data (structured, unstructured, and data generated by AI) in different locations across multiple clouds, and you need to account for existing and future data security, governance, and AI regulations. To protect your data, you need to adopt some best practices and precautions, such as: + +- Use cloud services or platforms that offer data protection and privacy features. +- Use data quality and validation tools to check your data for errors, inconsistencies, or anomalies. +- Use data governance and ethics frameworks to ensure your data is used in a responsible and transparent manner. + +### Emulating real-world threats - AI red teaming + +Emulating real-world threats is now considered a standard practice in building resilient AI systems by employing similar tools, tactics, procedures to identify the risks to systems and test the response of defenders. + +> The practice of AI red teaming has evolved to take on a more expanded meaning: it not only covers probing for security vulnerabilities, but also includes probing for other system failures, such as the generation of potentially harmful content. AI systems come with new risks, and red teaming is core to understanding those novel risks, such as prompt injection and producing ungrounded content. - [Microsoft AI Red Team building future of safer AI](https://www.microsoft.com/en-us/security/blog/2023/08/07/microsoft-ai-red-team-building-future-of-safer-ai/?WT.mc_id=academic-105485-koreyst) + +[![Guidance and resources for red teaming](./images/13-AI-red-team.png?WT.mc_id=academic-105485-koreyst)]() + +Below are key insights that have shaped Microsoft’s AI Red Team program. + +1. **Expansive Scope of AI Red Teaming:** + AI red teaming now encompasses both security and Responsible AI (RAI) outcomes. Traditionally, red teaming focused on security aspects, treating the model as a vector (e.g., stealing the underlying model). However, AI systems introduce novel security vulnerabilities (e.g., prompt injection, poisoning), necessitating special attention. Beyond security, AI red teaming also probes fairness issues (e.g., stereotyping) and harmful content (e.g., glorification of violence). Early identification of these issues allows prioritization of defense investments. +2. **Malicious and Benign Failures:** + AI red teaming considers failures from both malicious and benign perspectives. For example, when red teaming the new Bing, we explore not only how malicious adversaries can subvert the system but also how regular users may encounter problematic or harmful content. Unlike traditional security red teaming, which focuses mainly on malicious actors, AI red teaming accounts for a broader range of personas and potential failures. +3. **Dynamic Nature of AI Systems:** + AI applications constantly evolve. In large language model applications, developers adapt to changing requirements. Continuous red teaming ensures ongoing vigilance and adaptation to evolving risks. + +AI red teaming is not all encompassing and should be consider a complementary motion to additonal controls such as [role-based access control (RBAC)](https://learn.microsoft.com/en-us/azure/ai-services/openai/how-to/role-based-access-control?WT.mc_id=academic-105485-koreyst) and comprehensive data management solutions. It's meant to suppplement a security strategy that focuses on employing safe and responsible AI solutions that account for privacy and security while aspiring to minimize biases, harmful content and misinformaiton that can erode user confidence. + +Here's a list of additional reading that can help you better understand how red teaming can help identify and mitigate risks in your AI systems: + +- [Planning red teaming for large language models (LLMs) and their applications](https://learn.microsoft.com/en-us/azure/ai-services/openai/concepts/red-teaming?WT.mc_id=academic-105485-koreyst) +- [What is the OpenAI Red Teaming Network?](https://openai.com/blog/red-teaming-network) +- [AI Red Teaming - A Key Practice for Building Safer and More Responsible AI Solutions](https://rodtrent.substack.com/p/ai-red-teaming) +- MITRE [ATLAS (Adversarial Threat Landscape for Artificial-Intelligence Systems)](https://atlas.mitre.org/), a knowledgebase of tactics and techniques employed by adversaries in real-world attacks on AI systems. + +## Knowledge check + +What could be a good approach to maintaining data integrity and preventing misuse? + +1. Have strong role-based controls for data access and data management +1. Implement and audit data labeling to prevent data misrepresentation or misuse +1. Ensure your AI infrastructure supports content filtering + +A:1, While all three are great recommendations, ensuring that you're assigning the proper data access privileges to users will go a long way to preventing manipulation and misrepresentation of the data used by LLMs. + +## 🚀 Challenge + +Read up more on how you can [govern and protect sensitive information](https://learn.microsoft.com/en-us/training/paths/purview-protect-govern-ai/?WT.mc_id=academic-105485-koreyst) in the age of AI. + +## Great Work, Continue Your Learning + +After completing this lesson, check out our [Generative AI Learning collection](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) to continue leveling up your Generative AI knowledge! + +Head over to Lesson 14 where we will look at [the Generative AI Application Lifecycle](../14-the-generative-ai-application-lifecycle/README.md?WT.mc_id=academic-105485-koreyst)! diff --git a/13-securing-ai-applications/images/13-AI-red-team.png b/13-securing-ai-applications/images/13-AI-red-team.png new file mode 100644 index 0000000000000000000000000000000000000000..61ed150c711a49b24b123c7ce9e7b0953ed22438 GIT binary patch literal 173763 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¬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̆Q zl2vds*L62$VD62ebsMemF~56>CZ0qOY=n!5&SYGZ^5!rzj{E>=W_YELTuhLUwMlUz zw7P<>#dyO;SRlvBbAgV$aaq&FZ)GI6F9w>z(@-8_E$cGL8?OOJR#DDdujFc3Ev{BJ z)8079is75jvoN?tjxtK_KYJqw1oMNzH9A}RUz7ACp624y%(%AQX>EB5L2m;ECN0?l zM%+lobCZxpGwYhk)pgYu32k5*ZS6#`13RqSfn$=VXuu;L`~#ZF)b%)ZYpE`7E~FE= z`?9HJRUdQ)89WX;RX5zLxbmd%z~mG4qTR}|W_J=sj(`oLYjd3U2lD<^7?7|zJk)VM zTVd$K2ix9|x#xkLbM>kfi?;UDT`Yze|cy_@#+mDp5PNJ3_UV9rktfjcO zg8`CD)q^u&I}@;sZ`?sr8>r(TWPyx~0o9RyMpMWtdJ)@>II1Id zaD8!(^ju&0iRHl>#StpSe)bqH4$^r5^TvHMoC<|p^XhugQd(+Vj$8}_fIE)nt}Fc6 z_RVFZZR^Rak;(E8%l!Uy(JdMg%2yoXy3IcE9^5<4lO`Ol?V*Usa_6bY!0HBA0nKr0 z*!i25JqM+0=-2W?cCn?(Tb3gdx{?uxAo0&pN?MVz*4;tdFMk6xeFqZ`?hj04&Jy? z>7Hukv6={k@xvg1xC;lhD|f!RkUxfou37gB}9F}=hjmPr)0!;by4&rea!Y}@LQ z%L+*p-bKR$9ID6f4x}jTeb22aL$=10>Ro%=oh`l5O|r-P&oOX{2MDA1kETsrX^9FU zK4>7KFz@~L{*}+ccRHCAqQfo}&mSKu{{RAyQ;tZj3po+7KAZ`Xapl8qe}M;9_N-bu zRE@-!<~Ze-3QukaPoO`YS(PHafvu)FIQ!?QHI)?6?LI>Xw}1@#vFtw+&*4`fEtQSW z%*QT$dz17XD=S#&jIONCb5heKwRVk%Tw%!TSywi$*~rK$K<6DfuRxY0)NKks4b>Qq zN2PK4os6(ZV&rW(=uSWVb-X2WmZd&jj%MLTHpEAoJ7sh-RU?ZDr9Asg7p4n>Z zX0m^@D>8_~GL_xNPjOi{Q6y~JRLQqJM%5k2$5Y<5O&By{h*_j3?;LZ3jlB0Kfk>h| zVyvglaNH0`L_WTx_v}8kX5_oBdZ;-750|eze_E*Vm24h?`=4rHi&tJE)LIzV&ts6^ zIQd8!9Wnmz^Q`xrUQDxGM6w=pz+C%`bUxJClPH7{mCiD8-k7Iz0N-)AU^-{km5O?g zSegrjxwPH#HhzAf`fw`M#Cw`sES5w2%XZt1ndcuv)9Y1)Y`c^a8Am_u4#uoBUVfb% zj!E)cV`Gz!qx>m}8BxnF8rtDn;xWSpUzt9JohfZSqiETMyMN0>au4ZNucIu%&>t~J z%uoEaUth1SRSYe`@=>I=lZTD*kblp5Vd`wft7TVl7| z-G;(LZ*bCGKqQl#ney1rIbodj^{#_X@TG#HtdnKLUT5#33bGCkPc4ztgT-*qJ;GV+ ziJLbM<^KS{SCjdYKPu6;)a3h9MZYNQ$W(!}WpFS$bpoODcN*pFU-1T?r`hXrOQXuf zylFHElBEF|+B4OMx3zGdGuCw-YxftE?ub8`<*I-SdYpAUo=t1rLT3T4-M8Ad9Rbho z*!BMafU(E?2zpg3yGU-3A~1PJ2-v}4=zaY^D!AI(gKayb&*ZdW+L|q!(z>V9bJSCz z(l<6S`crX6>a#GdH0P9n1#7{e$Z&SoO{VCf8~!5eHX|Tb;%U(Y912{?6KP2EhSMdDbF|i0p=U1B!LL`j(_%3m2TJDd z^yJ>lfsdtBTv?oyqIkxuZ8EVu;;^KV)ww3UhgHy0OyPTaSD$NI3qc`d99EN~8AkCo zp^4X)Mg(^b58gkWYFcUxgmYQnWpH{_t8U}9a&okG$;F*5zOuo3)7x4j_^Y3d7z%8P z^+p_u=Ro@75%;Q;+SBKPYb#^aO&b~+QG`u(*QOl_sUp?pmQTEE86j%Z)9p!=W~RkH zRCT&b%QiVCx|>^bv>aAWhYBhwK9$(%@FU=HO7<{uxk)YDbQDhu2+l$2T~u~4?HpBW zyCm3fI#tF>=2C;P<$fZ9Lant;dFAXbvK_#R_U%(ojrRg8gx54g0g8@muF6rdl`5Xj zTAqGEpjG+oibzH}R9|Ib+PLKgnd(Mqq9o31O3WR?tH*3}$gL||au{S(SS2N6q0 zGOvGT?Cj4+w)kVH{kd(eP)JDw&vEh|dXH*vi2f658Wq?}>#-v;hZ_JKAII9T>^03% zq+4dZnW07DcOEg@G;7-B&Z#(IEUMu`FftNOIL|enBBf7ix!2vd+-ZiT8Bu(a^LO9; zja1M)n?Y#|&_Jv*?(&VyYLvE`b)1+tkbJ!f%!mH@10S7aKpHRKI}^1@$vNkb@u+Rj zoAQ!z-54g7A?JaTRrnOg-%13XgeS8J$0 zn9(dZWMzQBNqxpR2ivFeu6ps3lD@{+cNUsSex^U#OE=w6efg(-qDHhB(T)c|d zWZJ;@9V%$yGcYIR9Z5W5y6MrTmoe#_67IppPYaKZxsBCzb5w%p{YItaHOD_Vgo~rE&|%uaFhMJP!4C(orR!q>u~} zdXzm6)E@qr^s07tYq1MOW;N~ish&1sQFG<)`=I*!f0K;Uqq#B=AyBi9IRk4CO{cXr zE~E|;(Kd6+4(He3{{XLA(AYz4iKsG^J=J?3Z)4d06{J$w%5B7#Pqrx=NqG^19DyX5 zh1vh{{Zh1-m1YXU;grR18DgWjvv))qVUzDYFC0aM3UKg zmKQkKJw`vLKDDn+PFC5`_;X5*7*3<9TlsI%2^d%R$nWo3Hxs##v1wHgSriFSk8mm1 z6Rc6H+xhU@pu%lU>5OzEt!K$3XXdx^T0_yTZakO&0KQv;T(gs#xzSDvoNd%e5;pzi zw0S*Dx_q3EtVf?-YdZa(X`V@AmNCO!BySt0dk_ax)9|Igl0|lV^mZeX(%YC^dVI(T zJ-XFDwWRlJCA6<3;l^%l7>S2cNr{io(DcUxp0!bpwIWjJIi{0nONi{Ql14jrN+MPx zt8N$ru;;D~Qe+oQ!%VxmS)3I%^16l~^Nr1b+m9rA4z(PU#H)zz-%pXi1hR&Wha)bF z${b|l2PLY!pKg&$Ug?ouTi|@~V=<@8ouztU6fBRiD4!WYKUerDu`ASl7f3V0c)VbtP{fV0#t zw8?=PhIKBA&5@jraH;AE1o2VaT-&reFQ&6x!Wb)C9E^?tc>0~H2PcCe$j)m^e_z-6 zk)YWY8%WYim}b)^P)FJ=qUVWmxgN@rfFh${L=CHF3dnsc9EZ`Bacd38Rfp1$F-M#$Tlv}rM~YV za5n+owwlm~35jI4{y2uqjgds4u^fUOnSdYx*Mr)#9whP%7gIq9WFdk>f_`1 zkZWjJ!Fz6zwvds{B-X%r7ib)-N4fr7oCY5-Y+-o@x!b=XbvUfA7D(0&kjTmbWFd)h zf%my%j(QPS;j!~p<*e;(uh28a6D-9a3JV!Fmd-Y=eLB{sGhb-D`g~308;8`^3#+2Y z(m}s4DEsQ!`@k{C>5Ow(w-#3`V6r9Ry}#K^%M_W;-#2am#|J$9GtPLcvBM0F<-?b+*f4d1Lt$!4gM1n=Ux7?$2yZuSqj6QM6?NaW= z7i|o?ixp|VcX0q4SOFU|Wt9OV7|$f*(;)MKn$BY+Cu!Wjhg#lym?gMLZ{@PQRujy2 zMDdX4Aa2^L+yXFgGuxV{saeVP80J}St`a0cmbYa{)q!ps(|K|4$*I)oi|EYXI1`pY zPpIJ3@m!|Tk}DX`;#B&N^UvgJkyjtWz)%|pu0FLFXy2CR&7H*Bb@jc}EDJ#*%M*VI zAm{SqrF79vY_>XG?9u7)Ec%3#VF_Y`?j zoN=gzNCdmpnlKp~xF_EOk6~J-na?OR&ef>9YjbKh_9{{SjoM_;y+?9=I%X>T-hnWjRBW5{fvL4ZQw zu-bAuRw=jwW_ZrRNgyUbT>4asG1RfY)})SSiMK@MN#NtZvGuB^WRU*=alJsmKZkF_ z@~5nFk&vX1UdEbtDWiWzz0w8ZHos+X@Mq?GCkW%;YXYa}Hq|#%jo^k{@IfD_>MK21 zp!wuM?YlqAH1)VdAc@=&(4wF56!tUaWsu&PKTedgO7gjBS(%S^KKcH&ucvsEPF*%g zwJk9&ijWrQLl~6_^~#XrgZH!b6_#eYor$u!j!~YFBEb0n0CWML!i4XjoNcJpT%JiM z(Rrx?x{gqw{{U->q-2$tt2~U$)id)Cz|yNJI4^_f+*1*t2n1!ueHWkpwOWQi%0hM- z9OVB1oMyA52oEjCwOzJgk||z5AmHH za!1TZ0Z7j8ra7j&@0E&z7ocB3*idH8tv^zVI73V$NftmBOc!(n^2oUWk&dMI80(R- zV|3RM%KBR@K~yU}+}UxLJ5oFjOL3k#Ip`~n2iC3H-gz5@tL7%w*r)hdfrc6H!2Bsn zB2LFy6k_*LwbLU(9lY@)NI?aeg1ajQZTi!?>))D3_1d?WEFrLH# z)cr?XbgJS>F0Qs(D(UdZ6*D8{q}}{S@y|d22NhO07~KQ`_jCc5<~a8C;*?eN2JNal z6u-2%NQ5afL6WI69LqEHI2{H@8Rr7El1$?&W$37*yAj1==~jyr z4Y=%OU6ZRE6i&I%Vo&gnarLc4EeM4mE}!%4iZLzeBa(eEGtFZpuc5qeZB01jOLzsr zD=nS5lW=7?Wg5(vT_fuYqr*G_BpwMS0Bfb`qw#mVf($-yQDeFDre>O2D&2CMdgH)!0$r%j6h6mJfTeBl1uM;NePC-@~8TPE3kX|j!>{(7m2h%lXW?jM- zOqBx-(-^4Yj5KVxAKn-ha$xH6E3WWh68U7*29QW;ydRtq6Igv4iEzq28uFp^K+-Ym(i>X7G4<^RgsLRN2sN{Y% zoTXyp7bl^}*l1GTneBMA=N%8R;MQ!GJ7#YtCsMOP;SsP0}^t`R1|XwU$_agM}(2Db!lILO-+r<`LM`g&HKhlb$Pbn@~x z##8bakRg=rAG;a#C$D3lUMb4kiAMVzk(2#87=ZbsKu!jEKR5Nq;Z)Vi&A4szG8mk5 z!4-ORdmTnocOtBxIb_M)r;&r{^r!{WgpI=zP~oxNPyYa{_ogm{u?Zs74AVv)EyW}X ze)66O{J&9}j!h%%H}XSqGZb7A<)}VjJm;TLifp$MCCf)7jH zGFlN6^PiWhAH$WVwYOoYKQOMJV0csI>TEYcl%qs+<~fE-O>Ek5q_Yh932B zYS9%3aIYYnTeyx)2Q?&qB$YtRAI`D5M?<37?KYa+apMd>I@FKENrCeZ&b%@|6v)5C zYQlK6$5u41nLf^F_Se_rWj`>i*|j*GiQF?@PyYZ3du=0eq>IH)UnPxFX$jMl-1UDl zIgpY?a@yPx$XggSiw}!fuGJ={OZU5C&7NyWP29qC`k4}G%jTR4kZGW=0<>K|Y}Lzn zcdtsO9L~lE_IJ1GO>HwhtFDJiGGOAaE{Zn&-_nf*%_D^V%k^4gX_)O@Q7Dl^r z`21_kZ|;usgH+b?ML#Q%_}4V!sohR3qu*`(VGhxkADt!7i(x2m;=D3FQsjTjL;2L- z+E)$ML;2JsIY9L*pAkhX95Ael>#=gAFFCFrE6GcYv96y;Z#Nr{KJ~O`W^+`fWpa$R zC7hBv(?*@`U5eNwi^!r0Z8fGX zDW=nolxQtV08-+ty}4f130zYZ$eW6czA9H46;?oKuv~1?e5SE4i8-xD1l4D?^E-8` zj)FX53pZjg39B+`?YB6uMzZjUd~J_B`c>&X80B(f?dexM91^o;(9C`95#$nCuqWlB zC|PA(jB{R{b>RJ=2Vp+4{mAi(` zO?w&R8RHn|wlh&%A>Fih=CoeQc`d`3oPdKT9*2tQs~rg|sAZ>%@0P+V> z_z){gMu~2qlF~)};02)vu5t4Z#B=o?^`C0ljCqR04uo}Iq4f4VR-J%nm=SRzfzQj> z_BhA2bi$q0k;zh{TC_E0EbSf_3-@q3Q?1#dzY+jk9PVFLJP}v#bn^O)Qd-2sT!nQ+ z9D;g}`{*2dRfUEtIE0ZACCKD9G8dk4{xs5VH)KjlJ0frF*tF>kvyU;G9#RAMff)za z2BorliKS_$`Jy~HVSqvGN2k3qAwBd+jxj8JWw3ImzI~5>dc7T#@Z{UDOM%3QIRyIq zA9~U%%To!-Hcki68hK`D3t})hD1ATr-;H`Mn_@JpB|j?N$Oo9pfPy^_{{UYVhv5A@ zWZm7uvBo%U%ehN>`+uBQLw<`MWpN|`gZ_~T+?e+rs%p#ZlS()DpF$amG?)NGFH-jN z1u8w!zoljB-f(a9%atJg(RpSHHhN)~Bz8S|)R#AxFh#yim)8Aq6td$4@OqD?Yd$?v zIh-z<{#>Mz)~@3cHgSQtVE6nhB^I?dl$4NMII&eKZnU*;xi!4Ld@a{Ja6#walUbLM zT+eRz7glg;QM>F(tp?XDMp*DY00(gOrMy%2lO43@OuKdpVv$dmE`NxEFradKjz%hk zlKp{%*N{!7I6$d)Vs|Md3=9+3W=Z56n%*eocUlN<-e&UcbqH^x189Qz0Ae=}lO!Ik z&j*0Lss@_Y&EjjTX|(2M%G}z{e7D*`VxZ?d=Lc})ik3;*?ippdwzbt&A|y(TB6K@Y zWR3|7!1T#AoglEc^P{%ZZmbT-!&=?IPnq*)1Y;P;-Om^!tyPKiFG+cEG-K^|i+2L` z`(?M9j9>SRf>`o*5Q@D)oVOcL_DDta3?U$K?tfa&v|~s+wu# zZ{wB_u}Qoy+H{Hc6-A608)kEyH(mxqRT#{-3dS3&JN?AXJ-zA(-wMtATpW>>J5M}w zLKf)1J=`{N!zJFEcWCUbD7m)6IfN;WGC&w5pQ$}_?@=963D8F-vjxg40?B`BNqlW$ zH-Jyf#ewILgFULOmk*@0k;-)q7U5!zHiPCPxj8J^#?gkz13coSf3)gs(X3j2tZwF& zV{|JN%uA+UZ&k=qjPwN6b-100BeX(dk{v!XDghyFI{YhxvE(NQCvdH#O5)G*$~VG^?)M@zeo zBr&=+;Ut;dTeb$);~urBvo!I=2imPQojp9r(HNz?;uy0g;KwYxi)YmB#Uh)rLeAz} zD1?7$ODm5)CM3t^M(vh6iREyifGdOR+M`*mL~#A5?6PbmW@|=51YnRiN5KqOu1`F3 ziqMQ++FB%{ZBE`uA`G!yLMMpJKN-ouP|L_9bNbR9GgDN3KGIoYm5f%OWdWh{rE&|9 z4+M>>t=+Cw{eTq*_q_X`_9ctuyg~DD#i`$HeY#8S1 zDSqr^f%9>IPZ{BQ9Mvn^kvdNI@y-p^`C;}&!2@W zeeax+fyo>LiW=SiOKB}^G|Ap8SVx*)E+ky%#9p~ zO0uxQQac*PS{<-=LxiqBi2hYH7jj0&qbm=YSa3yC$;iiAQp2G9ed=ygiZmF&%F-i9 zR1Q!9k^cbJ_ozJO{sjL3zKuZ~rZ)@;81$(d4swGBsZspL_{{Xr% zPg}Ra40i+9KjT$%MrqtmD7>y8mE`(&{{ZXK?^DvLi9b4(WDWQJR{r;|*ZgXeGjXwX zUlxviTo11XnxR{j$Rn`rQOI`L#=Qx4Yv|G5xP5%IR zeHd~-m0O8zu0Z=e*@^!E=atao@P40|6^pxk&KG{sp18>Vb(v>*achxpGb*=L;H!O9!hlZ-yGbkjyV*bm2V;yj4^}@>T}OlbD{-h?2~zMqt2XWA6_7B&48RgIp4p_= zyf-e7uPC=KC{i9DF$pYpe525T?@*+(3(z)iijy_i0H~n;Cf0ip$(^?rdZ4E6V*5>%ymK++OP7V41T3yH z%ud{G10C~AB#|O3Ll4=lkol3@w4PE-4=hTYu?li>Pdzv_7XH7l>*Q7Kskx{A?UrNy zq6`5L$NFbrPCYU2-mlz6E{6x(0NrtXxF7?FK|lVXT>9K3*B0$Op|286(W3&8+bayI z2svOt;C2`$uiNTZQ19ISeyt}4)s*3L&OqoeImURYX`04L#-+Ss-W9fnVIg7&QPNiK zJ%FtdCg~knxfod-{{RRbzY|y%lR;=+dqUI0fUf1U_H*C<1W%=E`H`xr_wfND^1}sv0qI=b)KgjAG9-mh-cm+G4}a8F<)noE?b{{B=5Is(dVMM# zTGU-yiAT!Oo!kl_?Cfb#b1R%~p$IGXm^Vh7zGcrt{uPfh9j9o)ILhO$c&?&(qw|R@ z@=t;bJ}}FQ%e?zLC^qq_EIwVt@_)Pat*&ykj(Ov3D;{>>eO90lIQ`xUD#rwUYgT!s zPDDt6fzbiyy;XVc2MsKX$4rsls^pd|AdpNqsZqFKu)xp%0IIzQMU*YPE1k|$lpVnS zD)U&23xWHx#QoNDS|8gp!yrpn;NT81kUO8R@~o!0jcL!6^jrG_DhG_d4 z;B)Ek^{nF-_cxT|(QZ|qXl-5DO3N~x%!)}==fC%{RIYVviQ_nstWHA6zMPNKjMZDv z@-exL73VSm>U;eU=xVyY(Yy?+4XmcR4n?@jsb;#A27KM1+K&AsU+$jS9gpGZRBe33 z4+OH1<0ETF(Dw$XkR8PvxI0g%rce9Kd87BE07;I6-`Dy5dIe(BBD)HgnPHv57-QTD zzhELaa>Ut4@07&Me(D}JpU$h^!0Q|g< zN>L(n&+DAcGGXD#cnBDPAQ=X?Lk9X8;icC zjg3O6N$FN%fr;rrn?{v@F2cIaIwQ0kS39R$kid~$W}E;}4QUgKtGO$~kSjw@kCBl| z5lE#HtQ><|H=)Xwrv0czPz`8Ch{CxOt;Zh)fW=R1;#jkd!np%%*a8_SU}ZmY0M8wy%v~R zM<1nQ$jx0uP`5%@S4*WUD*`w*R@+P=#zk~Cc6(GDcB7QWbrUtL?nqD^WLBgujyU|Q z9wpmXBKE4aj4qj{_H|*>nr$)2=9>(xxbIhDkz8}ZrOd;XGP+w1n5l)QwO@(={Nk)7 zwrO(}?99Zr!K(=tD@AQr@u>s@(x&$BU!)-;PCeX45^Y zHn0u+J!$ik?AgqletJ{V*PfNJJ&%?$1wx1gHzHI`4JB=M0(RBQsV#!>jw{J5{8=Oj z0h9Hs4dVxJb0=Qetm;lucSh9cD63fPbq9#6V-?F@BDB0?9@Rj2kI!$Iny|BpLdZE& z-lj2`qe%?BiB6Xx$6~+As3oF9qBDOT$=W~NAN_jNk=AWHVspAoe=23x$md|+>TM7p z9yEjt-AZ{H<9H;#a&g+KOAJpZ+BRgj zEHTj5KF!pcchKvNPiF^x%=qoufZDxlO48ag1a|~CryVI3Snz|dwONfsL~rE!3hi8! zmb;vCbE?fNJt`}EfK;g?1o52z0FhpcrTCjho5N3}NpEWlJ8ps5a#i>QH?t4&dwE^e z?1Jewo)e%00E&iViLgHFe8UHxYX?$QrKW|dP^7)>S)<}8tYXw;j!C2}58V;$1L;`` z)`7T3Z>)#<3e(EnMkBgLkO75$tiO#?Z8Znnm}AhX6*SbL?#n|-$}(419}&0CRwOv* z3|!Y);fsq_)F3Zxg@HSeGh=A`-D{SD&P~V4DEJ(J+>gSyG}w*3y!U%{v~e<_C#h^H z`rsV?RoM--I4fDdck}%ZTh=f9J8h-Hnr@!6J=_L8xdFEv5J@BWdHqFsywNN-1y(6q zLCazypnh2uRK3mI6UQUSLV)Y^2OU0#HFnEQb+c=^CboqWw<_y;!=Cz|KR>XvF-*mwIO%7=I&K*Gw8F`#@OX$1<<) zU>$u1zlR(q>>@1xKBeOfon zy!^zEU@^$4CcB@^RF3%D#@{NJUq0L~Wev$39Gs5As$NKz_gON(JCsOrSET>B0IEcAvVu)n?QqvtbGI^yiGL z7%npKODFL$1aL4g2R*83=D402E^n`UVn6F4JX}d| z#?M){)Z&U(xGQ#W7)Gnf57c1bFMX$ul_k!baF;+>Dp@`jG;4X;OGJ!o6|OJhIwZmTRvz-Hd+5g?t0 z(}UX?$>*ludYVX+O_bb0bTrGGILDW9YUX%uCh7oT_f81U)3q_`h(+QS^7Oq;ZmnUF zuB5Wm2t;Cma5*?~c^Kf}R8rW-1*$fspj^Vn&`u?{kbuha3^@cGE)L(R6i=nxEu>Js znQeVE(#TTk)D4b2pj|3kyWgux_cN_&gLN;lf1H7 zT*!^ILOjm=;Pcy(G0i;%cYBh7HPz4AEOodqW13kYLw&VKMMNdZeC-El0O$B|Rp*Dz zg>SCBJ`p62Nes4yco?WDC(9YY;Z!mFas?K! z>CJTr({3)(Lp`i(8q6Y)oMu9$zFr@uNXADVi6XrJ0I$NtL9FABZ?n$@!UG#Ewfed{ zki6io02wE#8QLn!MS_;?WpO-kw5=jr>M0_ZScYA24lsGoIUIJ)S7p>LG}yO8E#Q{f z&Tehg1Xj2Mc#}ClHU>{k#B`?I*a<$?-`J@eMs6f{t*0VOa0;m0hbL;c=abJA2LAxp z{4s51g)QQI>|Ke zw}bp7i1_XqPAY^_JX6hYVXI~(l~~(q5#T?U8E!$!j2x&P&fmgVt*&fZ8QV*^wP+xQ zLvyEG7usF&k{-AuWC4-SKE;yLz5f8O$NmWWiD8~QnG)Mgy~&s;j;Bw5~0QdZcEWKjG74iCRfR0!ZfY;P_#Xo|ra2?Se}7Di&h zMn*wgkUO2xcE&da$#b<>5^yt&jG9e%D$d$}UvSu@ zhSN>5R=Lz|S+`2@ydq0?235g8NY4q*cr1!OklsbUJ&y#lk3L}=$OrG7kDgWwmK#(? zaq|p=$>bYy+Bv!MAhET%A859KLmaNG%m>XF1m}~(91sso0y9D5w~jdEdrQqV{{VPR z)w>r~{pL6v3@YV#Je+blsCG7s>~qqUSr|fDlO!uJ!6V#u{OT}$D$T`|kjm<+WkozP z5WxN;BlW7J?LTuL?^Z8Ev1*5dnt5V2kb4S?%s)@1DQu|xsi3hoM4a#+*PuV(f6u)z zQIEQjpKWH{sj z_kO*H)})RYqHpw@j1ko#`H$z&ezYb`i*U|V{_v^#uRo=2+bhPaH0qY>IaZZHC+NLR zVnwKkIIw{rKl9N1@A;meg*|6&{$#9PL*0%|XpVU+p1I+Fh-sqRtY*=sh@M(mPV@8~ z0I2o<02oi=36WELg=7Rl7~3AuGm=jv{{WsV%2K8?kC$@BFcK6~W+ux|$$mXu8SdiS?vgRlA7Ujb1jm&>K7j9&b^%(g{ITfEg zv&a@rL72rHgny)NDGTi(qICdyl8}1z_2Z(Z^C1$3Yp7KO(lQyI1O#E#KICC~6*(iP zLTI)B0I%!%@)@TL>m9lB^Bu7VYAb9rLf!X~zCs(&bBuJU;dv4KgDXKXOg-QQrD8Z# zLD&#RGg;9*kjSq0324$xBQqljuD3AADhCLe0~p3LinR=k*8W|@Nen7fy}i?UVi{Zh zT7?9(jgCm^?Mb2B+_Jne#wE9h$#xr4PEnR~({b&b0#Bt`w~#}*U|}V!f9q9w!yc*+ zbMIIdmr_e{=ErnIvoAV^!@l4Hjj<>2W2Q}N+RPdRc<}^kPnT*B`o`{Ax@Woksv7>k zuj{!{kF73Hc>$$-H0G0?ny-%fUTTEHh zGVNvgReEkGllcn4qpA|2Xlg;Ftt&O25u$ynK$zI2oLM?if?p|0mxh)rYX zt}`Dbk;Xoi$jH&0$pXe*v${Ub>PNq|YMs^0XC7Cfk$D@JBOpwJsK?AP*WRcwc}jtj zSEhf6R;JLoa!N?23Wutnuf0*ZU9aZCcB>9p54q|p$SRXCVqd7>lhH*z$Vh=gUF7P+ zfCr$cq`6rXFO|P<+ywrnn-7>ST1Pq9rb163W2ftZLlV@Qd!HzX0pdOfp=0-t>5srw zYqj$kZV4lITz`0V{)Vfw2XC0?9XYFT4Y8Q+hnE{;4IvAI*n!mkT`0ARy~x`AC2ahn zayH1UIGA_OzrW+xB+;JSLfe326aC!v_XpfpM|a`7SbQoqV>$B{U5X=&pEuCtk+nhN zI5m}PdcIpM%ArWX+DT!L;vVCn9l`vPYoOd(dYN)S@UZh^AsBB$c+YR=HJAoR+=TAI>N|eF&Yq&vp|#h+WF0~EJqY0Tr)x`UMAo`_SGV$sjj(q%Fx$>?)CK~%v_|q-!th%i zKN^}z3v`0KluAxP_U>exUTO zmQ+V3NI^tn!h`;-^~d;oeGOIBErZUNAYfcB@-fl+F#iC)fj>ekM8&boOJ}>aRed)- z1$3T9O*QR+9Vlg~v~ta+^U{!Mhtj%bwG4Vx%Vq6RWM8#t<%XNIjw1^BQdxey$MdReioK+Q>d91{+pQUFVI3Cr~ zS}1r2(yPfBS%~+oRJ1U)j1#60O1EiXI%2d6t|?%^mZhV|SPJ!Z5D>mxca7YyO)R3=QhI`ay;EGXfxhR>FN3o77 zH2~u@`Q{A7gHbE@HKBsgoni*05C$DNsFG2RwQ3gJYZ*3r?P9rZ z?2lXUo{MfQ;neSuCz4h_XSb8?C>$9d)MtP@oY!^YOA`--Qv1%j^CaOMN46u6px|*_ zo}2#wFt}s;hTa?x!zcNg_0JVr%KrclA@fMuu#BKwVq3G%_&BgM<9Qm-Fv zRbEf<^gLxInmcmxfb*pvXq^EW6nz(E48WKuFlvToy zLDrERoP*DN3W_2cAWv1}@Top;F?R49jsea~#|YXpj=gBLcZ%Qu#ASAkr@#3$lAFG~G&ow~-5G8OD#(6T2b})^vfr1ba5@B1Uu#iYsNCCz;kXQp`~5Lqk#j4-2>T}J zc`GW)xd^AKp4iCfeR!>^H5zIwsmBY>t=~c$sU)?9S_W&#$to4{vnRIh{{TZ=rN*Hf z-;rT+CA_Flm?K@TNaqDQH%{2&wP{`6y`)mMLtIKWg%Xu4g?k^m!|Tm+GhN3kG2tB&`3c?z*rc2$qWv_0q@->~FqGZ|}L_DbrHV7CPC!7w4?)%nc zI+{GM9+RlGn#jnlZ+L-`zHH>2V`w0;By`7Kacqya%s;a3KeZJ<)x|UUb4P)-Ha}bv zGmH)rsNIMq-43HHerDVw>JYRtE?0XwWX^c`Mtfqqqi^f_{{SP6-ue@X9?mtqIy9E= z5*bC!gybrc0gRl2I&yoR!=*CapH|FK!4>SQ6BKFTXr*#UUBIS6#@zFcr>#-AvimDB zn%?Tl7jK?RbLXnA;7`mLXK3=q4|Y8((KH)TFNFhXm#HqNa=vZUj@fXe^yfVL)V7ki z?Ay@gVel-M66RQ?vxOTHM>OLE=bSb{_8fcWx2N!R)H+>-u8pbQ+xcHSkTt-b9p1Zh z(39_3_dXugE#osw{>iDCED((>}dxF3-dIABd(9MP+}a%^)nE zVe?{MGtU6@&%a9E(e({((mRtK>48!)BPHm_Qp0rb+P7>*xK7qeJW2inX^$Gazs4Tc@BSwV`gwj>)xbm9aipl zg5KueSCZ~;vu{>ccM&R_uJ4!(05=R|0xBuuSuG6W*4hFbNG0CS=j4ndpd=|cC-{0A zj_XYk&2hfeTZ=nDUR^qGv`4(-499WW4sdXIJ-b%dOo5ZM5=thz(c`t+vMj3v{#j>z z#Ryi;0^|Y*UI*yHBgw`n~&ADtvQe_wG<7O)mbRqr9V@=wTO4wA{{@Sx;iW9!JPQY<^}qHR^B zFvgI=*=US~q}<9w54sNQ^uYtDH53Z`{g6#woA^SA5vaF4`vD?{ramem-+|=r_ zTf*->fR;F+kV!1f_K2E6FeHXvSDmUzKgHCFpx5n#IJPZ`tOUrpVGTz$}D);@h#2{oc8$7fy}mUDgtX62!92 zrc36iSr_hw=jS~4>ye6qZsL;G6@5-KkRATnX}HEm?!tz_B|>4lbl_wPG_{2k{{XM} z7Zx^K60-fGYZ;a}Nt)W}WaY9++6Xz=dIOMnJ$RK`Y1$ie_AC54MzOIGOR_!qT#RFI z1e_91SDc!rCsvWZO)~9NG1$st$P66d{HHvX9OnRmjw+q}aQ%iEW4B18V$CBh$~a(2 zzz9BN#xQf9#b&K ztu1ORC^ZQou(^Wn1pp<)5xNKCuGi!rTya^2Nawexsb;a(C~%mcyolfM9Dgcy=*kSl zdj9|mfBNbwa38xpuxS=Q5gzWpfHd%kdP=`Tkx`GRr5w(J%ft)U>YsN%?(OyD)=aTW8u^5Wi#QUlO6Swq z`y3jk$xER$mv<5{ON?Zs!I_p30RuO z5=iUvksF-t7&$oSoZB~dj~SXvo2zJDgH0#dgdjYFY_Ufe8D?d{UO>P+3arvg7XJVu z(JbICEycZs%NwA|sGF;M8+pMGO%|bQDHFb7-zr zBLrRd*8A+Rf6X+v3VovqyRouRT;-#4_dQrVmUS| zcXM=dqV`|_S!3j5cTBE2^sZ`4>)0pqZtX$Z2r#mc6qMRn{{XZ`)1OgVHkXfkDD#AF zVpVmIdc@2SafkLC5!JZjjYngl7j{o^9AEMI_Rx|;8cXuWJ7aNk)s8_V0gp=DORTFj zv4eXVWw@1-TTx3Qjr(%;~Ksnn~Zvc6k+s@aG2M`yW6QR>X3jGs;KH1<93{ z0Xv1$sblZ$TDH(y)RQfY1D&UYa(u?HV8=3L6!kX~s!EtX+$kMfe( zIU^pR6Q9GaXGwny79fa&d7NTWb1%LD;YJ4^8@$L* zRztVyK4JWb{A#)a*$P1JGHKXpn)a9RJSGX1OmiG;4gRhDy#efhz|?mZhTdjpVp2dx z>|=MnJ$*;5P>D%)Ne*|Z829SGuOE$HveZnWq115kFdaJaz$4s#ql#MrOooaR8RspZ z(voqy<+#ZJ56A)iD*erwMZ+D)Mq+i&7%XF*p+XLF5w_HdoMH~!L6DiAQo`RA)ZKkfu{S8N+^azN@im7yV^sADw6`yeX zxTb)}zL4`=wf`jGXs!$tG4U9I(e{WJPg(A4JLMO6q>`8 zopO?~kt9S((x-wFNWiX*JTY-IhS+~9)3)%G;~%>}l@1JtS~fEDyK)G~uA1IIlBKrr z?9l=8R)kt>pur!dRASM~IVlq9$K)$fUk$TlRf|ZBb6|>HLKA@4s)?J}%OsxkQ z_n)n8S@?-q47GUj+&Z6?h^vt5ml5=1P&Z8Wh_%?5fxLcorv|KrN6q@zho-wY$INR| z-^BAGfs;#7ax{-*8mxWVm-|r<@qV@BYvP%`Pv=hm0EDV#?rErzsp}_FjDHaP>IpSu z4Z{lZv*QT|U*}LiisWX%}1PgwYPBohOVgr}jni`hTjX*HJl$S(DbFL>b@1dit|LWyn}Sp>DXJCBN>n&JUAc3J0D8)PZj)5 z(ls9p{fA5QCYoglvt}@TG1k6&(ynLI?&G+#mEfK;l#v?*S8;LUol5IOTw3YS-Lv8E zAhlGL#ygS5dGxM~RjWsy*MD)rg^R<-4l?Qf zuRqKDDJ`w!Vg@_a#+r)dL^_SO_907o?c@t_3&yR+O7KV4oKcl?-CM16P{-uP1bGDg z?s#v*(zUb;eHv@Q4y|JHdCh?}sS4U>D zcllF2ao5;-*QEH1!Ztd5YbK9!m(secyGlULbIPF3dLF;6b4NwJ&gW9c7~B|f_*6zw z`z+F2+epf|kIlA>zyd2KH+ti-KcyDdvvcM*-UJ*sQ`)YJBMdtk8wx=@AH)3n(^A+j zJjlo+9I@n8#x~T-RWGMNo;f0vlnwK8a61u84%M9lC@frZK*+)Ns{%A=;$~#q{?qiv zeYpUbmy8|D=75aXkt}1?(%uUMN)7Z zo_!B*YN2;>uu3GW>A?`g-||<(6+g-#G|dc>AORe<99lU0Cj$%~>2Et23pqJne znN;Vm86N(%gc2cnAhEcbIaISXz056^Q~=?CUZuBZx67I-N0k>^N>4(0e|IQ4=ANNh zM%$=P!_iMw;QXWyt8-2l+LWX!Pi?A1#AX|Y4d$`KZRZ4F1HObsM9GJLM>{KiQ=MKo82N3*S9P z3H9q(c#+g4rmB-D1gsZnfwUasoT$i9ai2lWHb~8kin^AYC9^5dnFE{*V<)&L@f?~8 zxApy5{%2z+jP*S~Y%JF{ppm&Gh&ay~IXr>$6VuY4FT@K&31}md$)ba7e`m;99GsJm z26|T|4ykt}n=N%?GdiLbxZ^yo6DP1gqLWtV)wM^4X)iRpYl!6z@-%9@SS-ApjzHW> z-}l^9hdwpa@um+Lw^OtvrEfdL$t;Mf(~#-183ildgSl*Y41cg7rJe| z@pW<q)$b!m#8spT5nLz#yDrnf6$2 zCwMfiRw+K!a$&^aL|QR|9gAQ9Pi*$7{M(ssr;a}mTg|oOa?^rVXADY#xqpXo&Tu;Q z$ObcQBPN)2jVpb%!F#8k&mL^CoE98p4Y(%+aayKk^!~rhWzxwawdMYZv8=YTG%sX) zwOq-xNI=NIJCht|BBFcM`wi8#uCa9;v9*Z}wCYHlf~26w00%6?IN%=Up%PoiV<5NH z=Zv+#?=77eXZM^$4@1TuCj;@Tw=v(`$$H5>hNvc*8!*wJ5=uZ_zFLqRXB$ZXXPot+ z_8QQ&KlYX7-mP;srKajK7~wNW+c!*GZqXPZHsQc*VDNLE^&3rCCDFd!VR;l+7VRUw zyZMbOWwOdUU;~k!*{H8YwY~M)>Y6OpQ~8kyT1KyY+KwyTtpQ?1RNX& z>Bsj*c*Q(N1h*v`#@rDSJ}qwo1|C@_8?p-%f=8$$9MwaqJ-wZRHO#2-Jd&iaNE={m z3~j(FeqMRO2cYLKd!5#y%$65oJ2egrX&F#o80Q6?92|_CcgAX5wJ(+b0AJD?H3+=7 zx>)0aXyc6$VUdyd4m`nuj@a-4^jAvU7!Z%x1~*OZ8g=xL2nbwE1W7Kl1cUz(Z}J-T`pML#JZQ*+aQZeMHAef zdw@Vs_z};wT8nDtQj$)`FC1?)f;g3(+kmPvPvKWm+TPUfykRqs`C%Jz{{RAU{Fn0; z(rc@4sMxj7hHZ_caL4|eV$mp-0P;W$zi>H}Fi=`P<$r+o+tZRL$FDS6z@^NK7aD}O z8&l;-)BRj%yHP%a`ka&AqD=X6x&HuVe_a0nN{p`3GuJfoacT$moS(>l_2>^I3nvA7 z`%|HwNM+q^BO{THIuYo7N3r6okJg@C46hrzwL61F(Q<#XxmeU1%L86A52)r(nVj^Rvlv~0%(l=k=a9@Nn-3zM{Q-k!Pc zaZ(xJm9cRblKKAtTL9qwW9k0@e3A4trrJf1%|=-v2bohPN7uUj4l0oh6noXnRA-eI zHD`A_QBV?Co&X-Es~}YXleqiUxFvCNk|d0b7-7R;k9^f?XCnRGh0UCBMRjX563drS zB9-2t?amLO3@Em;pLdvIWmL;ExF8_iw{gd9#~)hWf=f6;#4hh$va!h1?uZzfnOp(1 zotVx+=Q-fjm`=-do_Rt$L$)Ov@Romnmh!{XL23Q z&V5ZS(VNkn_Oo$$ELLA@d0dFwY|=&)NH-PWj#vPpdFhN}6%D?vac=>PZ6uV-9=KfK zer=z?R9CU$O0}8)zbtwHU)N zSZzZNzcyU4Y-26J037koYL$iIK=HHrVEpR+k$}!X?s&k@{{UQML3!lb%@Kw-X+njO zv$`xCoxJ0ohamf6q>4`Y=BQRW{Wny^?~VRMm@_GC9pByu(>+I_?~Xg$Tg7k!vOJt_ zxft0VI%6Dt0QNm=$?Rocv?rK1E^xzkGCsYlxYBNZ%Zo@avF)0}xLxe|VBTZ|1N0#O0PMB3&U-$lRGDjY#t!Z?-l3G0Sjc^> zqZ#z%kaJd9L5pK?8vgZ;dGyEB)n7ak@=F-N8NwU`#xYD`A{h}RwoJ<1)q<2Guk)yY zBr-_*iRvU!2|l3HmC1%dC{yx=P{%pPPtclbPcV-@R57kN%xX{JngadGvKE5oIktV$ z8Av$|gG`n&1&B~V$@Stxxu6Paw~wjMhp+qm975((i%9_5*UR5a%#j# zM7tYt>_Wr0KaMKp!5A@#NhsSGMn=X1fN~E%Tu>*brR@YmC7iYzgKc>m5Gc;^CI}b+ zbU50;k&smX01j(EQHmQo=ZF=WR>PHxA~ghMKs`?c9@x$)){z9ffZ!jUc?Tz`t<5q` zE%Z6y)*+2iStGP~$L~aiKsf&ZWOY8|RNmT;;?U*}=S%^&o{B2Lv;s7JqYZ*j<4<&G zq$oi3$mo44bqw}utU4V28LMOk&7{|Hz_6%iQ{|8kxgh>W^sLLPXN8JLyI2wk`d3e3 zr_VLkA&8ir3cK`C*b;q?58+svhnm+Ctl%pLY^WH)&on2f-q$iwFv6yMg%lO(QnF@0 z!iEy)YK#VIb-BRg3bAc^v&b-bt9Nq6<*?i`MI}FYa;BUvaE^pI z9Suzdv5<;==nHh%>6)6}aG+MQFp4f)TZlkEolgG%G2^vE9J{*I+n2EBv$?H=*_Ehf zjBVnrt_ty0W>)p$nq;(O-Jk&1HL4Glv0X*XWapaaH5eF!Tg1z$#9sm=HAis_T(;52t-OuMQ~6bx z?PmVr{HxV1JRcBIlpQ@P`uIv7fd}PQk};KD$C9nApncQ%Q(oFoKIyMwUk4%S82qX^ zd>ny))ep+GN+vF>mqW$mfyZO{R6bK@p|4$^!jXc1b_d~Emp&3q94F~g(lm`-H9X`3 zD#f`w#dMc`6lCQEW# zW2vibRL)bjF;QE7?lzGB00`sjOjg`?P>h=9W5uRr$MFu8xVlEoHMI$FpD;&J)V?#* zws}8+^s11@n4_GJyZYBzrRr9;=pOYJXcdY#?+gTfBsq)5hNc+w8HLx-n2w{NsaBTz z8ef-f$yZRjlG#H@#KJNGJy*Zvd8s1Q-4iRp?kcG1=0rJD$fig{3>jB)Fv^45{{Yoi zl3Ehmd$U^KQEA~=uJGLCC>Y0~`qa_bM=XKinFt@iQ!L!{_xvjlFEb*>rHl0Du19?O zT7zvaZWpA(_wV+u@-S=A^aoGBCSdeOx*Yd|CU!~o1CY!Wli9r{-E@1yKvS*>;4>vapZO9Gcjv$GJ2 zj&RPSfI;-fTH|KaVup8$#`X01cWrBVmvSiwr#-z-UX|188lej?VDJOOCeYzef0Xl% zojq%d)vXft<2L5r_{Wu7Rs|UkJgDKZk<%xP);GJcw=I%pW~9a?h}^s?WZc9(x|qlu z1C`}lJf6hh)k~2I!d7W+Y~$gCkjdx5fUz0ll^|qexhK}4XPO&;kc<#lbTiF>$_YDL za2yf|?SWYDcX4HQWwieQLAc^cA#gw#ZOQ>T1av(9wW5`@W^N6uqf+MV&0{pTcQ%*o zBg*cRWTPshApZas+yRbz;CHHFvtM_D{ma?uVWWJVFgZ9^+ApgKDe!7xtWxt zpuO(qbb6#saLK7l^A+=6-%^Ln5;g|{Mfvi@v)Et^lYvsCQd^_7l1?@g8GA_u#B$A? zWP{KUa6QKv!LFmk9um{7G#PbWKTf@~k`fHI;I`EVi~>R8xOGTv-xECgZPYt(3}j`? z5=!G62nVM<`cR$j7Slzi_5F1xl{A^1eCT}0#Gwn@4U(9Ver7SA?ZMAD!K$!X*-VS} zZCpad7kqMpxnlnSbaucPCpaByNhG+DBZK=&(qkeK`|*N+sL0&If;NTzU1|rC_Gna3 zX%*bJs-hA=XTUi5an62F57*wCvl_W@r->xDiGY&n^+_1U=;h>a4j2MCu78)W*%ZO9O-~4l#krAmi4kNp>`r9aHTZ zb-PDntVwY+GDjR`WN8^=Ohns$U;t%OzM%IRs;Vuch9h;O+{DnuD?ISXNM>dtHo+VP zVUE1u(%f44GRkf=V3Ao`g_>PHa>^ui8zgtjFC>hCjGj2CP2#SfCB3G%aFHx9D_csf ziB9a50AyiUmKn!W-krpweZQ}`dfvpg(OM?0bYb%)d0>_TR1lj{v62ZO?aypgKepMy zD$it_qf-{^Tb3*z02dr% zK9un*kK!lm&-12I(-TOUWC!mcKI(B#0UM)Z*@yM07Oq>^ys`PxFoF#K0IhMs_5-UQ zVff;xmaHwktoIiQ1Q9gS$iO6Vjm3|k0=iod3?ki&d*3?uQ0=B^*pfm209_>IV;;Ro z?~1wMTZ>G~@PE*x{jmnS&)13!gl zj~N>P(D-V?+eS3=^h)_b=A(5r`*jMJABnmv4htI&J=zow%_4T zicBI$ir@@qVj{=~wiM^`r0K?A9IpLr3Q}4^Hb*2iTf?b;XiaxJ-IovcmynqO9S(Za z?sUj?z1KQQELZ(xu}lS&{{VUs>;C{u{{ZitgSAa#MZeK**56Tp#O#65z&m79(SiI3 z_8@~7d?olJ#r|C&A%8{?40)D5T>s#*|YnoN<<;D&DqAN7OV9u=S zn1AmuEC_C=oRh{m`AB)>VfPgNgMcbc7Naw;XSwO^?@XYQ#FIcg(kPI8!`yzf;lR#* zZq*uto;uX5zr~CW^+b{kZzZ`o=~f|~pqNaOI!HPg&JX8MY;xQKj1F<_QYt7vGpPDx z{{SkaX3Yh;x48cRkBB_C?<@c=eyR6=Ewy4BDIfQ+6u6B}OCkHe?*q#IJXU;uK*sMe z{kT#6YW=;^M32%f*2$=H((QYk?$=zdU zyfOj<=cjI`t#&s0G!_=%G{quwgJo;-F8q&{Sy_$x?(8A!$mD(xd5eB|?yj)tmgmVaT?8rxWp&b?MiE+n6GmfBf}i4j2=q8w-ZUD%sKDPDMs@5?K^S?>~IgJe*^hfX8sXNINQrvH4BMT_Yp-4va+A~ zY8+s3=to|npuEye)Nb)K7a8-MMvRdos;Z(o2+3bFamLZtk9yac3uBHeNEMl|s#xHh z0q8PoZsrX&W}U6su4R%ml_9r{ytsdgRBW95xF8&!IVP!U=q)t|cpz(7re8WF-<`3z z`BWV9fN*~t&e)4BCdYcL7t3r02OY&)va`9om7}wT=8;!+#2GlxJbpD`&7B5I zWYnV<)@^wc9llbzQ<8YWAIsjm+gsReWHDP?1cFo#Ci#(sPxpUzbAizH$*xk%O17{< zxA#74vD(pFjl>?jW0n~AUVfF-z>J9>+onqx2_`sK0fFN7RvzA58xw(KK`fM^Yo<98n%{K6WpstxQY2$c`7lF&P=<}&G$<*?Onr!4% z8B<+LGM4Bm0z~5+R?bYvmgg5F1QWrfG3GH{Y}#Z}jucjVX!vX|Pf9tH6N@>}FbFjf zOpZNkee?+nladG0nATBRec&=XnyaB%qaHgKkb*|l_p2rdpc|hAj^>?ls3K%UHy>Wr zhkbVMoz0(WXj-wl#;qJtu#cYA!Cu^(iHSb7y{R#HjdK&_WT~6b>!Ag|QQ}!+8V zspHhHm!zhs89rK6YQpDNE!L*G>lbPP^9lTGLr>SHRt8okx$NaCdu;9$T!#qQ^b1cE z%!&j<@v9MQ_jgwyGj-}KnX!$SC;*dN+FqMHmqEO?+w`llMh~&JRLY@_b5g|%bgfHW zGDc9OVzq7bg-%#x`_SZ!)Eh9Pje#5*mPKKMT^O=K*is!i8~AA^Q;E)`<#CZuXIVHk zvnG)l{u-?MR@0Cu&|Iv^lG{1OWZ!AUfjF+P*%coCm0EjWmzC{T8Q9KH>~KO1EozA z%fBFj^rdMA?zIV))sZ0DparcBJ_V7o)t0gh=Ogp23k#7OjGA)TP_@pU3kdcPlzvp% z^ytc;mYD{t8Bj7RcGM#))K)3Goh~qF=CvI-BN7E=J)*Z^T^^xuk}2dG$0TOD6S_Gi z6q%DYnH%=1Z{hgl{t=(0bX!AmYFLysgNm6=#TN={Pg9kX!pINZ{VO`>OEzqH;=LMJ zR|;#N*I{2Ip5lm2Pev6fxN47|7Tu$eh8QB8!mr&rO?2`oz+=*?%)5S-$f|2&(scB> zBm{Y25Di>m{)rd68mAV*qX6c%?6m8PnC;ylk>lOfS2;eO{j0J((pwyoN-V87F{(hn zcnr7Jq_(#M=aY!X>+M!2)26k$kl#lnFT3VR6svtOX&scV`3}gGetyEHQ&Cr87&&N* za7pIBPu+oxb{^H2EzHL7lG*1Qdk<={X?&)AyaXd?UrKGv09}>P*U-Xu?a2(= zea;3AHkCJPgkQDYYEz5%kLE<1Wa#QgUwX3^jdOUX%7KbA96mS=>7S)*+S=PK!nU88 z9Y#FaeprG30A!w9=y?^dajL-v4ppvWV8wO?Qp41A{V6($E1_~rvBh1&)>htJFtZbo z&O7@LYQ&Ky5@te1dSo6y3hVWs5yyFO(xhTGC-;dS6@mOZ*E@T!U9&1hG1z|TDUjcf ztz{aDa<@~Xl2DEM6@+K5;mFQLde!@OVQ(C=?cGXn0%kY ztVOHKyp2Gtp@CLLP(D-cJq{1yTc+=0347YI)Qz}8r0yF|3k(zUw>Yk&!f`TKT;0nD znF^2H%A@9qx@QBL=ckpK&e1LQ7);LC3f!7_JjgN4eFR?ycrOy&%bM;ok$6 zKPlww0E6G3dg<>kEhO`15Vx4&$@@ETASVD3$qnhA4l!J<<+O6l%B2ttMdr*zLDbjNt@MkVH%Tvk&hTKh zz2cq^DoU{A{M$(YXCAw1*V!eUi`cF2rVA6?2{Iw_n+gWkB=1$h#(CrGNjx)6Z}w@m z4Lc#Ci4nmqy_68d42+NfE#KCv_S}aX?V-l7jq;MqBy0vgMGOzVb6&0RqgJt9169(l zCPcY$n|%G!*ckkBKU(vxLh>6K-^u$loz_LQU*3HK0kn4Y$TdJb1W=25!s53*dPvs^`~3k`L_CrwZBUU zq-FmAK_$qW=HL#X?d$Uq`PKj44? zc+{+9lsm$lZXghHLG|lQ^KKH}!LOd+#Wc+-Jhn?F@fZj&Pv0D1u5dA2x=L16X)xh2 zfJYegtz9DKVSbHbCXVP3w*LTTd@y!wa5|n2Fe^yQp^T|GtCJ5T!*cf5(ShZuF&N63 zn2=dOsrcB0fpHtMg;(sc;5zT#p1+}jz7Vcr(kKHHGbz}N{XlbRIPcn9Ad|(sU zdI8u}0^%?E=O^If{{V@p7HC-K%9H42>-tbjOzAX0PlYw3+s(CK)Cv|*FgQx;vf<9uPysHg@KYq zPhc|Nm1z%$P;Xs7tqF9z#iVP0_3at_r$b!BaJzd0&$+J3#%n{S>qX-tFSEv^mm@#D zPIH5fo`$(kI^7t66C?YmTmJw;IQ=SH)tbj*x}!*bRVN;Ur3Y#snSmp+pX*i{>Jq;- zxswNvGDCs*3{!sISBS0cqxwV}{{TYGNqUh9N#O=R-4$ZqE5|sB37j6dQfa^1m!s}n zG=KHULVxHWQy%7dm>K7iPJfNbpjKt+EUwPd@JOI2`|3tNI#sfe^^yhu0D5!>^37*) zwQ=fyrAVsBrCMy3jc9D+7RO|;^`sP{1JIFMt8#BE_$gCwZBZ( z^}UUyvMhG?@kWYr08zoL{{R;V$AoW`6;&uX?aK<}k1sm6bI_=#?B#dSo+t&y_KkR# zO^+db^kyI)TtvVV>$y&O`=Y4}2=B|=%`(aMR#o!tt+yX2Wqihv1py&!)nU^0B4oT+alq&ylkBAY>b7@ zdklgr(OaG+`-^j=+*=!znpytPqDY)XQq~lbFgE13-VGVf2nDxwIWBw|r@wE)$+x1noT2q=wxOKf8^*^Xe*DXPx}# zmP^Ojiz$q4wT+a3Cskgg6~+RL@5 z5oJb7f=JICQrTLY4S6*i3DVBiHXmtN$&lM4^j3Z{PDsG#f(=NP^Tik0E#`{Rk)}Tg|9nC+{SlTA8ydQslI_DgN!99Nr)6maT zM&3~!!AEm>E;3j)B8w-61aY+VJ*%L%&DFS>q``~kj!a>5(1Gom<>S0dWkX|RRwd;p zylfHu;1{>A9<`;VUA)&KQtfcx#N~D%-~-KNB+VlF9i6)imOKZ!3Wa42F13S~iP zE+>}S@gZ!~osI_tq-okc%n4(1*AKHU#BrXSf1W7j*3WRY(`#cDZ5mbG76fsPicjaC#N^hL7OJ+0 zd#S;1^Zt0KkwaNR(Kiv29C{jO+R_`Ah_}j47-o@C^EKeZpxgJ?uf247Y|*x#G;FwI zv}f_IPS)XJZQHj-dJuUP(dn0Quy9&847mfL;Dh{utmJJ6snNl)^%iyh5uQDTW9v`1 zOdR00=05pse@eD>8(VnhcJmRMK8iDn=XFg}?TV4cn`4baj(Ydc>rn*na_UK1##0_? zxjCgYfM&XMIdv>T=OZSs`D1W6>IG&ah3WGe(6nY^8TiTPHH6yebW)4zPb#RG06TMB zM~VT4ap+0nyDPi6pqq4(Du64C@m=IsB@Ap2J*m>Y!o93>Qq9Ftmn~M93{*^_x=v(g z$p-Xciom?GCC?SuMF#0Qb*gu^IFRv4tD{KIm7G)Bux^wkxz9@IWw6U+4nCA6hs-#v zTn@EYLpIY-dx@4XIVaM)jXS{`E!JsznxUpyi006e%Tyd^KfIIP58G*AxI$%>}lM$#K3{#Edu4g}Wj|~(<2r@+|n|oD2 z-nAmPPSvL=gPUhtW2Qist0amXk(`=%XbbERraqM>jVQa*)q0RHYV>+qhTKmzhcZu7 zu}0}~qaRVzqib_4OSOk#T=8P0iteth_S-Un#w&-qU6Zi)ttqu6V^Tb_8Mz;Z7^Xe4 zzbnY3=~6^+MK-OlljzLbjU-GC2kBMYMwZgqri`ClS6Lit!>4Lw@o%wo_N?P1pN(T{eMR>S0&A2>qz{;%2kTvRsYVjCOejs% zrllj>A=YDYkla(~qXYr|ri|$8n)~3vC_BPbwZmvH%hD}Zs95^T1yoJ1BKX89Kf&67T z$N>IzT9L}x?C@#{Is3yOTD=|gFo0Rd9`)hYzAwB^5aOS4;(bDOIh2Z@XUtlgg0+uf z)I3>dY&n%l>;+)YuZ!~Ae(C96Uu|)GblM+^H61!MOM9nW{pEAHdjA0R>$f(czy~xzz480ddh0Bw!t}#xOg3*PP8~EzWl-03I;7P(G%#YUf+*l;88uCHIhJkhBLYtV zH&3N#s?Jh&=tH9w8_RRLgIShn)nRR}jk#3ux9}WRHnVv^IL=v16kf(S}*wCI&v|NXY^6EJ!BP4%c z=Swx^zU*5ea>MtCK^Z;x{{RY*SWeR0%$Cw_$Ih~Rs@zqEuxO=aGD8R_JABN)=~_Ni zbH&N@DMt}=i+3rWvHt)c)84c!k|?;6P+akiqYL?tYO#wqKRJf^o9cu#rhU<&^Oa?JA;xt&EU%AJkW&U0dJJYZKd< zV7iSwmJ%o+1?69_UOIq!Ru+}5Y1(G5W1~eBeqWd)l0rie+2xgp#yK6kR2s&yH`*hL z^7W$;fFd`MFmxc3l^_+yJNL+~r&7s59Py{j`PmKC-LSWjudYSR^F;FI-q^(3ff)ty z&=u{GSa&+JCC`-wjl2WSRx(R6PQ@5tg=XV}gYCy^eayFT0B#_fWXrbL$yP>S060^$ ze8*|WCj<~H7JJJ{rA4*8Aci1Y-$IfC0*1+Q2UY}fGmKQmFR5-xZK(ys@?6`gxmUZw z%@~!=!m0pafTZID^dq?)DV|Ne+|dhbZ8RS?-Ac)D<2hbOJr^S<`?>4Vl0}Bv)I9Hh zXFFSli(npHnPbjD=vO1_Nq8;a&zJ3XT98&qA`&CS>^6)r#xuh0IX(UJS}2g@Ql84` zp;;x<)<~naW(=}!+NMGS9^;dYaoFRNmb?}+T$t_WYegj^SgwIPNx)`i1BC}Crg$70 zmS~{2Hx}?}Q`#(XEOJ3OKfI6;y+$%H8z-D}^{b@ndX@d8b5CciJoj$V$0Vtl7;RIy zk_v_jpk$5*7^KiCbutc+{!89j+<8oh^7(Q$#u0{91Rep&Y+!Tpo-01+x?DUug}Wr^ z#a!*$z>c7gTIe9Qx=TA_{grp+N~K~p@|;{u&cPc55O7x*1D&3{)jR7ei>n(3h6|ab zlg?(D?Y8-ZF)fG5^8)foI3qad1x<}3BNxbGH(S4$2Vublb(EYFVvhwz_C$ z9$b;1n1j>t6z?&J{{TpY$@1Og)9fPN+n_l&O;oKd)A!BKF_Ik=TDJ0TuEf<1A{7< zn{MG@nOmUk7JjZz@)X~+D*eb~ zA5|^<>JoY4kmMYW^_|VROFwL%7!rA7(G)N8sLPimzF6a{9Mm-7_YdGd&*eqKl_GH| zWh%t&JwWYE&Nwv|Y2)T!antEe;bKK>bmFIpvVX>_uUgWyf+%fG_2AtKw%Fp%JooOrC^^4dw zBYhm?PUP-wqYMbaQnhdr_umO$>u`7vLtQIOd&tZT9pk%-BBVJO!>RP^T0>LuQpT+MMZ)_wp=+Pd z%f)!+mGL9R9u%-$3GJ=o`zatT&9UCOIl<(5`c=X4dtF>I+v%cD?=6Do^WRxIO}FmF z`rrJ~Dym5?X)jBEk+<=N`%s@j(FOccTUZ#@SkF^LoEz5b&<_^qq^%U05E zBh#arD|v4s+vhlEZK&BN01mCwx4mG=v^Y7zUW_`5H#LpX2yBiyqcvykG0nyFW>SCU zw0|sv>)F2^ed#Tr|87ZEQdkYSX6w7o~yk?%^(yZ0g#<>%?#(?&gi8ePDQ42Ax5 zhDsX>2M*f9(ZEm-DespXK$r1aqDe4cW*RcTg73kU~yM5u^ zK2xL3Yja~e22*h0JWY?fz>&@~&q4UU|m6C8|4YK@(JzBvXZko<#@hi^*tm06YX+13%purMLSwSDv%C3aOXWh z`kKkqJZGwEHgehC>bh)6A};VocJ9eh!jexI$75W`wUk}UZr|De0BVL;wVHpkmslL- z;&Ff!Zy=xqa0eoj`$Bn$I#`Xo)cagOn9*64NSrc^o^V(k{RhoSQ;zl`RVLM&Wz?Ci zBFr#HZx!?`W1jx>?{#E3`BZ`dVYp;tI2_hxzTfm)h!Nb!BK^AdVvLF;7{Zp|WR4H; zH-2guWrEve_Quj_Z6t>6??&l7nAxymIpK(AA1E2?%|y|(4HQ=v)-7{y_K4$xb|YrV z1Cg+DPb@)xqa(1bNVUuQY=TQmV;q-88RwGQ%_U)7vS%?(F4SXZrT;Q zCg{sO%%3TlH)Xb#-~}M_$>NsXS*}U64XWL-23xsK*q6GvW89Bk;-R^-jyqJE?)|Ma z+1RzSIrC>MIFUHvMtkG1_o<+i%LB!Btz%~2I7Uufk&+dA@&P&H>C%}mV^I>>85&uQ z^on-*k{ysIus!(Wk4n_FxsjMEg7^S{sfAyn_8#Z*tazC1l1o@tIHUgnSnk`T#(RGX zgXjlQjyl$)b8Urxw5O0p!*q7Lw>zI!^!Lx!qs2F`rX{qN<|Up$P;MKB2lTFY zq_jGfoHaNIuZ$M~b`==fz@8W$4+rxehX$!!EWg@OOCy#NkC?IV(C77~z6EAJXAXJb z@P3sPlKrOHlla&7iceGh@A3TWMPofm<%D~ZL&}qkeQ6$exFj=ikO{!)idmC%vC4W% zIr{gi&bUpv{)EftEc_b5w+q zOCDI|rC^!*oS)DN&$_w+o1Xj%EGdY5?fqdDkRFuo^W`t9C_Y(8c7M<$0wRCuOwT5BVxSZ5NObT z&76R) zd(w7p#=XyqH4RSqd6Qsg?uyeA$s!HsKU(7~RfY0~`@r%Ich*+REx6C##U(qjRf3I@ zDs9auPW64P_BS-L+Jd8C$@Z--bJ&6f{o;@*+J(0vvD8!Hjgb7ngIW{A=GpGYN=tFG zoED*p7^a(WtQMlKfrloE`&**AVe{Oq`_|!>8MzNIOv^@fU5!<>DIj(N@ z;u)nZoMy3w{WrAEeS(;##QB-ah9{7;%`8XY=4 zx!^l?sEs){b4gW6a;-{V6Rj0)Rz)Ku@&$R!P&>*#Yz)^!tY2K;xOIV(ZAXc}@AxWfwXw1NT(=TgKzB#*CND_+V*jX#}d zIP-hRBJHC*=)nZXo(_1ey$?>(Md15mH3QrCMKIT}chU`l!&!CjL|?BRqaJ5TDMnS08#YwElS% z=yki4h2ytAUzmPkqXdsi)|!Rde=nJpX;g7ss^#mMt!@=K#9wh8$1PT6p4_@z51nM-=ObzKtiQLzaKGu+s?YxbJ)r>~>PJKQel@w_%@_M& zZY?ctVV$KL);U;2<%d9^^*-I}epyN_Xkw)ZO*`1mA<|i6SmF(WS1G$cohsejn}6L) zC;Ez^i}Q`h~us_G_D2gG9vr!U~T0&1POi(aKgb#4DYw2j1t3B&MTx0G(!})3;Ms7|C61O%#4Z4=0=+QIAjY`qu4&Tf_~|mZ>W6#5qt2 z{t$gnuf0^ZXnw|~U?aN!0G3R{py~4--1PMCNp&VI)c|PO+YT7xZaw?fu3W3loauAB zbVq673utvcL6-9Mt)q_GQyy)>mNmfO@N#k6(vw(t91m7 zky(BcwYg@xc8lSI_1wO)9%85E=&zdK>^b9eMP}deT?A z)1izKw5+uCX-MSZUj0oMww z+zb(bG1DA2D(=bTbdJO9x6zg*ZHO3Pyzb}bT#!o-mxamT)I-$0f-x9YQjvE{e#IAFb z$3alZHI$a*zLe3ZxtI}DP;sO?(b?}s9CBLjR3u9n z`OBUs&^ik@^pdXJwNK4Dcy_i11 z6ll95iBA!)U_Y5X$t3lrUEM5-w$`P-%LybF7&13^Vc7Z(!;X0-1v%t#>z>tXQcDQf zVa82bSwapzXY}kp8loQq<@`@`Pe;kcXqlTwPp9~<;u}=4)ZmX&uuc*g{`5osnaDn5 z9^E~wRtfwkaIGnuS6OlUy**YiUA;WsnEU~*ARLgO^*E==kt1??eJZFYjvSnfZ)Q$5W!veCGeo0aD9;dA;>M|^0CIWcJMU$PsL)2D=jigJj zUqKd~v^0{4!p`9aeK+(Y@;=mQa$9j)qLi~Q*x-k{%I;6DLIM8(pqi%mb}T)6SD|>T z;f>ybquT1SUt3&2&e2NAl|*luxNdg?&p7MrTr&%J&ojfvpl1GpmX<0lbSkWSkIK3C zH1!D!tcGSu6y$ka5%|-ddw$|+eXKf1L=`l)Z{Ma$f>Z|$ez>eqSy0G_Bq zzvI~cS@)o!E9O9}G$WJc{H#8ssQy(5ySiS452R!KC|k>vkVqJY9T+asKM_KG#mf2$ zE?hbC_QnA?C_k@SGR)hgC6B8d)UsT|bT?f?<;Z%=5XODV^gmIGjKav9(S(uz0DXA_ z>)d`c)rn)*+OAuSpvx+P;c`z*_s3jSCC7aJ70_w_03@;6N2uIGYa-@cGAQ#NA~Aya z$0b>bA1OHlrCKtUh@R3I&zJT?EcYc{o*5HjA&duX_1?Qf58>oig2J}>bXzNUqm)AL zEzjB=p}1VGLV`X(4oZN)k;ZX~@L$C&P~6X`+}<=~UECz@LV+Xsa;7u7R&S7;5tR*| znX4&kR_Qdz45sn`)<7x0KRU!p_JK-mZ!#8MhBkKja>S4`*kZM@%^}vw9o3{(a$K~F znBXv4!pf={n`6r%9F=T(1J3~Cpwgj_g2;xv7$w5a|s@3bSJLCoMQyA z7(FWB%G1>KlPV_%Xbk~ufY6R`u#J!AxTb>o^p(H<6=ZEbC|_@#A+^Nf zX#{ae4a!Gt2bUzMq{i*I{K{Ey+#DS9$f5|YVZLZ=-Wv&{x|LezV?`F9As;9bADiY? z+Bc3eeaMR1TSXRAO>rbTZRn2DW|reJv|Rb1D)OtncQ0;72AHj(Za8lhO@rE|Em5qe zg1vat#Q(_icQ{;M593zj z61o^JUg52gGSJw-BWZ0V#z}APBkjOElEnm@0EIO^Q)UPFGON#^?E2&(* zMnK5URd^lw&1uVGH9LbXjKA6!c+c2olWM{bRLI70yY7N9$v%~h4V~SXQ>jlLj}#16 zXzlPMN87X<=X37BY<=EX=BKl`ODI?DaGMJj0EU&%n{l22J966yAnhM7TsCQI(2WZT zdmIyNw3Y5|qw7A+kwepzTmT1boO^Yx4L-|rnXVkMo1Z<;1m}_mR{DD2nzg3tFzQi> zH46obLd8>1bx?QU6S;T?13Yuk4tc3=EiSC3RbtJ&Za0OC8Nb~)!i;)z`cci$PCZS1 zA_TW(hCRd{HZj3->h0KjQ@mjD4Jy>-UlN>UP7fWushUKH_O7sPz!+_55`^{9MZ zm(cB+5CVz5>2SixB> zt{3DihcV|pGsyjGTI)o(wy;athmz7E*wXx=A3UFbmnZ6dsJdD(vpK7g_FH$4q_Q90 zA50&+{yzM6rpk6&+m7ZQp5@s804!6air?k)C6)I5-HU)#ZIZ3<P|C82VOjq>&OB zzBj041bzqg287Crv$>foy1rW~LExIwy_P*k!OpoJb*T)#Tajp-$-L1#!`j^cAO}4bs{#X1!0-ALCf^HtRND`R3XD4Qy!& zQpVb0*UhXCQ(>Ou?>hj;NKPse8&9rmL zz^fMrY|98a5RIAjJy`x^)>pZ|Dpo_;u!fS%=H-616Sh$OY4$u$397!Rn|rM3G~=)n z>~m0BYEs!<5}dAi28&yTjx)H9MN~yH%kuM+#cen4qm~g)PUmB79-FLfk}N@TJ7cM> z+k0tk)8dH{u#>4QhMYy%NwBq6bGwQWU zUnTzhdr!4rGc$2W^BqD4SWg6TTqd9oo^pB3d&SO?4a%LNTiA?O6|ML#873?dqhtA1 z{7S`BpR;#on?~7O8jw2xNXWqIE2E#mmk8T!-*b8qS<~ouc436kk@Az8pJywb9^RYk zRD)c;wQt?be-l;@ifu3lF@Ks5|< z5^5aTN|agL*?8vW&Ta6Z&%0)>T=?Txzfi9>(A@&@UUVfp>q6dEEIw|6h)V2Jq>|9y zx|Ug2eZ|K43E@G`Dl~|w4&Q;SMq# zR(hQ-NvortG7xDDneAFvx@$u@80%I3$ljEt7MXFJTDens^rtB~=~mXvp4BvVME0oE zu4F+L?$xDgtjs#ops+rbt!H7#$fCw_lIU-EdhCF)Oe?DU3bcFEzdTnnXJh4naf4k} zm2s63KjBjsC_f`9B{k-F+Qv@h9V>goR`#FTFbze7a2Lyy&H$>B>E*Ck;QCgKc0qi+ zjAO2TwbGnonom{|=9f(fHQ{Jv^6YTISmnohqi&>HTrljo{c3xt$hlxT6H>z-EGEzK z5G#(C!zmrrSHlsSQ1c`(kVtWc!0G5}8=D#JFP&~N6^=KE|)jU+@AE`hLn92&{{49X## zCL@=XZ(Mb)CLwIuamg62snv8M`|bCfaitpaQQdBImtHs2uQ%LV-cJVB+#*>|$&>ee z$EUXylKQvymhzL!rrk@#k|UogZ}*2){5bX%jcc|!_OU)&jjT_h@Aw{RtQr}a8-_wb z_BE4>=e_qf_goUT!|r0bv72Ozwbvvq-PHSFWA&vNdpBma3D4Rs z?dSb$jW;@rn$ zo(9LxKC1cS@u}?|Rhtf1V_+N;5d%5t_gH>?s_%XGA;*=YSHp1IO(o+H%H|RoNdP!5 zqlW38qmVweX6nvMWs>S0PT~Y8(E{Vljn|)(a^!aII{hX@aLn@ky3bX&fmFP6jf~?60*v8^`Bi>XkP!3-omX?6 ztt?WRZ6<~o7gm6y%jd{uw%H_ka=6@}oNWQh=L0-siiYL28~81R*A};vYF9*tc8%r? zFoSa(vV)!hIO8=f+S5yw37R@8Iit;?DEOiIGv9Pv^#`0^{kU;?vmR7-!5B<^r3IybB=Lt`=k{ex_WZUm(o>l zTFGy1S+)yUq#?4nJNLKOk%9H1;bLoKvwP^I(`EY%?JOw@-H_-31UtK97|%GzLVY=` z;ZB$qX`tP1+arl!8>9|NE#?8v3XE<&`0rY9$S*a>H`)&n#S8)!f(0H_i`1JFu85QM%;c@d~Lzw9R_)+t}j`=A*MR^<0Bq$SnN_6?|5GCW4nTLqW<<=A5!@(3S= zNfpaOd1@_n#bkzQ5+<4gR^1O*DnS@*@}oWSI#rmY)MeJ)?C;~$E=|9f=Hw)Sf?JZ= z-MLt*lB?g2fbmtW+URN8b<|K;#~rh{^RSNX^Nh5}NhLvL0I&lb4)mUaDk*l-qXzcK zB8j#~BR2GJl$OBgTmYjyaC&vAFP73t8sqGijR%z!awV<7U5~~}0y0hzf=FSKPjO2P zt==@j9-nV=Zf@Ay#=BkTA(R-{FU^jhDaTxnwQe~s9`4#V@cROj-64+1d2K5)@}mk* z?)u}91_1-9k5c^6$`BN?u1cPSiertkkb4YPmD&>n++4+RB$q%*6~wKyF&XnSk@I}n z8P9I@TIvnjq|sZeFfx>lSLQ9x0D7L4gH}4@E09GLlgzO|o8*1~2Oj-<3}ljc1DclN zbcz#YX8X{PWt0qgtUW*39ld& zRy8&^D-NOjZale2{`!H>;xK->s^+>SLzU=2cMc9g9la>(dYTsCvSLx@jCcFP@uvk1 z{;B!?_$+vyst zQa0#r1in?cM_5FsG8Hio_DLKJbCI{XstIFf0>dTWp0@J`Rbt>Uks=2hKqqgQ-+Q>v zaB*_p)ntfV+i5p#5+rjmTbvhFJ7n51la}a5-N!lWTSn;QlfK4_8j z#Dh<~h1}Tb?`J4U8g~0zI!Fl;6y+OYkO%;Cjx(G-(%$HzTG_s*=PQV<&|4}fYlz5X zVow{u0Z(avk9Yxf`)v&a>Td_iIAY=WCM^7W+c(YYZKee zb0+H}_DJP`Ffo^5EO^**+1%mt#yVt-FWWcT$!d~3cK24O>t}CtOG5E6Pu}H8U_N(b z9RqF{0~Iq*e+1W4TTLa+-JHhq>R5u?!){B2j!4hmDBJl^7x;ZR6L!>6Rw15ie=bFy z`Q%G>ge-B}A!dsQWYW6i7gEZldh?9&Q%!4fvtQfKCG;(EBKgDYix$h^8PYwSLbh5PzN?`1-$oxY}<9F%VpYcZCTivx9$6?;GB?1z~hVvVv~K5xP~ZFVQmtumhm)= z6zeV# z>0uY#!BNBNI(jkxnW1Dv@=rThnJ-ts9?~(|1neE_>h2`vDWR}##%Y;%Dwv*8G zU*SDT2CYXbvNIXoZ9x%lSe6Ku8OBHhlHxfz3IXU%TG*DfF4~K@Ay{qA#k}#z^7#?O zu!XyXE8sEFs+cWx?3?&5|bEh3F=isT6-{m`|P@E-s)uC7VXINR;TDA3%P;}L^! z(X%KUq$wmLag)nAB+8(2-zm#++*{omE3id)7DP*k;%u4Hb>1Q_K2VCog##m>N(hkq zc$0umduQ?-%wH?#K>axcjyR2*F;}Si} zU|^C-2-|z(Il%R&xf>k~rknRCo-=VaHmW*oK)etY4haK+=rf!giqf&YyOCv%@5@AT z#nO2<=hy?BV;;YSOW=)0D{Id-ebP!nSxdG=Dmrd+{9eP4^8KI zjpXz4k55xrx9@VaH6M4HcDKt#z8zE(x&lFwI-dQ!nuevl3?ydbt4swugt7(9$r^-hSsd^Y_08H zXGV@__bA=CRZD1m*h!8ha4=g9wU^uu^*#OFrlN_0kU9(vtMVV}D!f+HEM?ayC$yGzZd&wyiWyLTm@x#`mb+N56+YEWs#Ze;VLornm;9mPuFut6P&_sweWautqK zQ<;9){{Sg$90GXYaas4oOKB6HHnS!=@xu@1YiCryyNs-=wyzl(1?-}H`rwt%=g6oa z)a1CcoOw2zl}o5=7)Z#+ZN8bHwcO=rBN>dcx3Q{nEQgms=hb*M(@TGI=WX(=KtJ^l zhyMVHsARvKg%1UTgP-|vh<_!d(Xx@uEv&C0RE^3=3Nb2hx#&sy;<_j_Yt1iAg>T?^ zN=I^g3@Yx!-Gdhr{NL*E2-n9k3qZRBc^KEc2L8(v6 z*1Fb*8`a#NH|L7){5p5(apVEVQC@Z%9qY2ub?bX=IU;a(F9x!zw9wXrw64qECbTl? z<<)bun&9LgHVt<7y7UuUvND3KI#(At9%0+lG*i8-!nK^Ph;f7R@M>j6xk8eFPh(K^ zHDW*9p?2%oRkJ9NL#0P7jk!S}aB8-x;plJlIfmcfMh*@uYffNT^Uf;F02R1Qjvz(5EU%7^rlmb4b^xZOnz`>r&D(%y2tZm~;S*x@iCvr4Y2V zAG;gd)|_ccYRIXGgVH*XLt!M3xQs4(n$@;GOQ?kMy-ijV%W!a`3+O9G&ex^}O({t& z5t>qZ$mlF3OX$3dk;xnrRqwPvv+gP~J5<(|#Q-CQ9M?pV{kKm!A3BajC`!)6PA8qu zr%94AOK+!`WPFa*dKiFQ8j#zsBvnPLS%Xh>dAYk^ zB~~%-ns}Rhk&K=Q>sc1YYl)UP14!-nQ5PiFVWapLRk@Gtcd|icY@D>SzaNLM;awE) z?fdUV-Rb`T4ozy%kGS5wmaN6NKxS`D`cs5JY2*&(yK8+P!j|`dTN1Qh^tjUhWZ(I5bd{Ye|oD!9%ew;Ag~8L z>#JQh)=PQ3`3iChvkp0|yNE2Un+)tqpZRAH$ox$^K~DD?oZLBiO_*+Mq41TYSBJ}* z3Eh?xgMe|+^~QS|!PIWj_EOQN-oH~;R_9E3M20)b2;==~uk_a7PuTapCPk^4bkT2pGdMjL2{oAYhVvbkE~d zpHz4kHlp2e-7SIrTbk-Eek1EPo*jKM3zS&pd^jt$h~uH{_*KTLTH9jf2|YWUwdCz@ z3fkI|shqToFwWk%JqMtuZIa>^5Ui#wco9*{XoxXAxKA$i3s0GF3 zj1XPQ(i4C@!~D%CsW*EK$t7abB-cZA_M0Pr(`cje*!_6H{Hvkxy@HF_%QMMoGX!HO zNXin!j20k|P(l0;HOCp}^BHDuG>)=4B@}ylns~aE9Tq88+%F+nKx5x06|$1Ona?Om zuXWh+ieB=Vz zQ=dxOF6`xTYAA+lw3IRs!KY>Xb6s}~79!z`Dvi|8T7>0?EAKrjlr{H#WHs5u?*YcgcAm6?%a zf!&iY6wVeM#F7kbrHBKL3Bei3&MQMNKkM`&wuD^Ecr@Kp$XU>-hI8eaV^A1|KXbNu zJeBE^1q%0;*E8QiY;92*D4~04ciW_(s{qBf1|KOn&t7^ARjFdSxYHfnN2tBzhB%y% z@+v|D60M!eae;>{m22fn6W^s_9HUQvpk6fDTZ*C)#ewRGU3c)n8#c&CVJOKH?Bjzl) z&)p{nCYcgMEazyi{?8*7S)eL^cng#A;9!BZNe3MCsqSFZE@i!Ip#-ul0q$=Rx8rk$ z3`*n^$dAx;>L_I`3U0c0*--_>(3y*QitB|&-Zq{xPE$GKudPk?Yd^HY`P8no^KmqB zm{0~WAl>rkARn6~V*{@_t1{i{A8ga7)NZ0Jc{<$vsy2>!B2WVS*vkwr@VBLBS$Syo zuWhayMh_xpMRhQ)-@HnL_p`SHCxe^{S_Z8RA7YP4lV$DHS4|R$EuB6|L2_^iJpc%B zc>CP&GjXK6nYB6FS+=?@Zxc@>+_d;BgiXUGmjE1Ng*=hg20E0w4EIvaYa}zIO5x;* zr@|8&01wWfU=BMGo+?Y&TF+LzhgQ%U-c+*+En{|%$a1T;R?bg3$p;zT+lrF^0I$fj zEcFSseJTsBFwH&Ht4RcQZm~%^Nxg$^PHvQAjBPXpz7n5(N$8!bi zMW{tMXyeJ@a0;@JFbFx$SRcE9dBuv)`uUHX>7pG&?DyB00x@ipNB;mH7~!|zkOdb5_;%_!6~&~=afy&B zO&pod)AM6&ga9#;+2jH$C?K>EX)@kv7bYg2MV?ff0^|ZdR3u?pbI)F*r9i819-AvH z_VCj-3#B#I^8v1gPBbB~#`o;d6Cbfs|F ze_e@Tvkx|qTTH!1xMu=eR|T$qY!57L;Df>B=Z=)iNT-H7ITq_vwhc6>WsV|JG86(- z?i}EN-7qoMtHWfM15QY;B)6XGLag(y$f8~QLx<-hgOiR3J!y3NsjV*E{>>ff8=_dJ z8B@JVoGyCz2eHLP?DZ+9XK(9rF>K>c4<(d)CP^7(^L~B^10y&Ca3ExJ^8iU16JE;& zlTPu$vhL)a(Sw0&8R~;P?jwWTFBv6{)wPBjG_|~uUfO00sd6`=R{h227XcX zq$i)P2kxyWZ`Oi^%{fUtQz^vYW50R;ksC5Epy@z6dGFeqoO6mo#GIOUCd0v{8L0Ok zT3yH1i-n2e<+h#!arpMDmP73cJ(GRumvny8C?tMkBk`=@bJnaZuNy~k5!)ne7#~Il zAl0Tskr*3Lk~`v`y_@ggkLC0gcUHGpZ<(T9u-qcg81z;fvG`<G|TCwo(8Z<888f%aBjedenYgoIlaD>-nj8v*xn zqda7A2d*BrT7*%_V?Fhagg5?T$rqfXZ;|r_ayc0wh0oqNJ%JB`=VM!5+gc@rQ(U(5 z9ExO-CIuQizDNKgVUJDOJwZ3?h+vx1C~Vy&wMin8KX}U|(<=Fm z%PIK|Fg{#z3XV={+FM*rlnJT7%XJKi1(1m_pS{|BKz0Rz`=f3<0PYsCX_iPYiv67F z$hUJecGFy#BZq1qHWIntmraTZBPS!CqM-iFds}#nc5%iFLa~oFNtIadS*3Ntw$j@U z2d~T0tXn|^y~J`$t4C(ZEu7wDR>?c7ZOM}=P6D2)PFR)5^pR3aTUEC*T*S=;@F2OB zZBhwUx6cb8%Qit=XBlJdRET7-Nhj~FoLs<)7HikEv{)9?&dO~gtHi*m*f0nran^yQ z-pMp}S25pBbn4b}+80O?O@sF`ZqMCP2Pcn|bI%bI9KU6`k3};?z)iaH_?EebILLIy zMiojv(mDI1o+_iow@;MZTj|5>@WhfixjX*=c;V7c3EbE_SOLHsfO(~%JF`aFJh;;0 z)l6?8hUyt0BPF3$^VG=sB16||;w1|fK>3YtMoT$~Ny- zRs$!y42re5n$XotwosBm8#)b_<|&)Z^Pq^3k0WO+S1yGB0EN$N(v+EFhSJpTy7J|U z7~NEu;9?|Q+R zBiOal$tn>P*D^bwo->$kNX7!~A~2`dp1eQn`u@KIDJ?_;Hlp^$UJEh|UUPu5>L?fj zr?1VC&#&MGUABQVmw95ej50{P%gD8qWbG<)TY-+4s`1RF?dG-^y}E=wlsE*svk@YZ zj4Kx2!1f(GVqHE~iKMx)X3Sd_@@Bv(##S|B-;KMm+tVP~i#km%V+uoa8vq0Di+LbR zCP`zo43YK!0P7C!;{L}`&|{k7c_h9gZb9UG5sU#|ZEGZPY`nF%M2*-O@^H58#{eFu zu5v4K2M|Wq@BaWF1+(UXl_MRoSk#T(#|iS>+_twdsWLVg@Sred&tOj=QL847w(!|& z0b%=8XUUO<1a3R$wrGVFW{x;sHNXuds0qRJEB+P9d`OlRlu~t(U%8}cxId1J?oJks8NJsH+5rO&|t8k`RmS)Iw07>5? zpg+idUbP${W!eejIU^lF=lWKTvk#0Sah+ptgL!Tf6>7`$?v=c9hO{q&{HmnxR8r{v~i7zkM55D0Es02GzTjU#LLvUR6d#HpU8f-0-dS~k_hxPfo=@w z6_x)0dLJ8<9{#?^-l|%GZK$Cn`4bZ!;xUYj5JzK!R)!>HI8x;GP;!3(T6$)dkAE1{ zVP&^`dKu630QZd|A1{}v3?o|H(I`pp^Xx<*N zwP`e&p623NmC?fzs5boMo`kVE`~`7uW(B(_?2C`Y=lWAl`i}N?GbXoVcmV2EK>CqQ zzl7cDh7d>I5g|PkFBv~TGh2&mw*Dqz@}XBeb{Ql23Z%A-tX+|sNpp-2iJnj5c>Min zdjw^z%(>%9ZX{@i)MJGQs8PcHl-G_0U9rdW_ce9w<{2^(9(xl_7RW!1-`PehBXr6? zX52pJJ!QnEy+MWtzKw8A?f%VZD!B=)Q93NmmzH>&->IS zOQ$Z!rY*rI2oxX8)yVAHRT5mxsJ~kA?K{V^q|D5?Qrs~az^xre;zl70V;jpT>m zVx~j^M-{0ypS|#yIjCi^k`gy5#ZA@hT8m2MPbeOOltMx8S0juC=A`n#=}Vf$a^F*! z@YSZ3qG(cITwAQf42b*4dt~~1RDN!+;_XRciG0WCu|_bFk3c=DqrokwhAm^2iD&9_ z`d3%**Hnv6y15@_l4&FZ{aWOJE4vR;rX^LxNnNJYS45VFJ~EX&ElPB0yC-JfS9E3S z9tpkEty)>6Eh3DvVzabuc3m!fAKt?M04v7BA%%Lzp{cf^3-8n2QyBTQ&Luo zR-PDBYNUq))1H;*;c*UiXF_XF>gQd0X+kxi&5sgze@;s@)-0SQtlX5x-9DAbNp^_L zu#K1f6oEO8}A{`+)gd@dNV) zx8;mw%@Yo3Npi@c3gS5s@q>^mg~O3A1oP-|T3%ZT5T1^>>)cjN;VCvT?OfHZ?Nd7F zU)eM)A)0gw#1>!<KL&9FS(HZMev<(y;Q}DxMoB?vL=OQCZj!x$m5Ir^|45`BuYve;6HDQ6^9tC^XO43iz8A6)18)nh1VgeNI(PIm^$2kBBG zln1J=F_am19*5G1ie~H#r%|(ZnL-wiE~6WJAIlYX>^Gew{jKnuqLyeta8z~Ov|#SY z!5{)_EL?e-eW~0N$$u{!hXXkWAbZyI+a$k{oylcq0oLf2XI@9nPdLtg@8ex6%TtO? z=$h~Om11RhzR@LuNEE31MT;2O7~^OwjBo+%*S#=#mtQrup>K4`7;U(Te9~N~Z@u@v zVsXwu1o4`)b!#*Tmoi0n3iJM-ZpkSKavYMQ<|U2;6YWt;6}{Ud#XK5;R{;W#v(2#% z)C2f(*$27fp7pACXFKkSw_0>E-|dn(WV5o1Y}0AY^GK?2Wy2mxCq20B#Xuv}=8kg| z_WGBZ95A$Uwp=SP*&2g_56S`C(0f&7ip0aR=Qh)jU7A44yE!VL4w)I}r_4PwRy?UL zE|y7RwbVhCq>~esm6cEuyR)6D2M5^Tn$c=Y({ig8Ng@{-RE-=H%I^_Vhj22>xM6~* z3JBbBobj5WB5B6ja*cBY!6FdCR5H5}^OkIk4CH}>{cBQ0meR;A<+h;-7Q@M5CRfan zxO`w?Spy7noK(NRp7{l|w))s>)l^BB5=10Y6sZ_t@iU$5DW7*c)*zoxv4ZLBP)k(=}cr zYLRJQ+i|OD<>rz}EOz~%^MV+j<7glrfxYP=xBY%4sBSfo+62TyYaBuC>@Kbp2-^}8 z#fxQF0^}aRVzai~odCqoXXLccD=8?mt0)7n-wH!;GlFx0kx_YY$!BblT`aE4D0n3$ zL<$Z@81c@2xd)Df-J044lUSP6+3E4y#HcMAaX+1s6@*XGuyWZQ2Xo!&)HV}RFv+WG z#!;jRHmXLpfz}fFP)u0k*YAA2ah@s&(?k1Kp0d87Zk|yW?y9cBPn#ZC$jHXtr`N4s zwYR^MUeKXiJvR2@SY!J}SEA4H>laupcvXDRn z^`@J91vUwPYj>yaMr(=w`Im6r8ALe1#yj#kpua!sxu7JU`%)K|!%(~8G&bm^Rs6_y zp(Fdp{a`)G&myWs$d)=s+B8OZmgUT{t6<*%Mq|!6+sMez-sw<=wp}dBm-h>Jg*U;@ z9D=|wBm1LoAMG4;tt~xbi&lu}+VP33+F7mVya6K$s2hbO{LFGO&UooYq_mgx{{XI~ zdpK?GHAj-yNVuNVvj`!Yeh@5d&z9O0mB?&k7#JYtqC|#RZD7fx5*VY9VFY0Fz#)6! zo_Xe&_Hzkzdvgukk{6OqtEZuI1DusP%(&XzHvNmEw^Chz2#3Ci4j57_3 zAv}SA*v~lvrV+KaV3LCKyn965-RpbkQ_91r$eCp8ap)`+NH*)*smmeD2+ zcOyGBsY2i=`9rG#j&fIwanCi1-nFo->7^-)Q&A$czvylV!m@zO4mRYBjCJ{aaaiZI zWh0`Phxlnfik!pl{V7m<#GZ@RpwK`z@_Tfk4CJ50(T+Lk>*+>w><3B)IP~=6r86G% zXF2*(P$ArBlS(iTA+ z=|F8d*m34wzrXI|{{Z5f^rl>1O>u~qY%$|;jwK4dyyLIqRV4(TU6 zpZ01(5Q>aCPKszIHai4H1ranaHxAirDP19k#*!NnUy_8KSmfP}Ne7Oy@54)TU z{*_B}Ic#e|r)aC>w3jz*vP&F$iXskEB;dAiNnAHf2B}rFh@K~g#yhzINypl*a2O&V zHYQd6^MkjLae>~bTScU(iN&qO^sC>9G65B%26uXYl!2ba2 zZ7teKG7Avpjt9z0gMv69@1P|zr9%p+Ac-Rj2*Y8MowFdF~G<;Bbte{@{F#UR-+{AvF(;ov38;d zm=HO^+#4hKPJ84_6mg}z(OXRfHo`cBiD_;2hgcM$sb+B8Wc>T%hYNUlm-+ZUA`8_RE)oQFB{ z2nPfn!~iX4RA~kW5Az9ki8--u%g_bDmyKKyyyZpes={=Z<(>*i_`l4zpqj?V}e}X$K^>i z+^=~QdTgX3!5TVX+lMXuEU3J;Iup*?j^)kFg6jDB_ZO*g=Ue%uwn&T%22~lx(cg|U zo+>NfDfHQIr}-hew~eg6(B^5b%E`Cpf8Z#fuR)NYW1Q8w?iSYO^Y^l>cDKgnDDB}E z=1@Rcf~m_URIUd=0pf`0CR(;;7Z%=8mQN~E98Mx1ypB*v&lpBvPSQ`ykH(!8n{@c% zaA1VT({p>h}j zVJXimj-j^WkEVGQEx(uMTkM4FDI9jz;c#^^fV+t7z=CpmRZ^n$p#dJC09(8mK#7Fn+K;Af7kW>d2VF)HuTxX$!^^| z>2bF2a*c?`?)B?{J9iboCz%8(1XnR#MR2OhN8Urn1JKtwZnsv`BhTeE&``GVgaKxc zAAgRA>sv6aF=0)+$#D~6U^r2pm?E>85jO3i8?%-nZ~<(2haGY2Tvv>47Ecu186>(8 zDyjg5L1GkVInFuuuHQ*2s;b{1-TZ60MmzggnCh3}U21EWwvKmc*~_swf19|hI-L+I z7mX(ZD4YiztkVO$bjJ!Y{(UL1SX?dk*kO`XWy-Ryat=mLKN0l?we{Z;YFfRWyk_3o z!W2!RZIjCkJ7AoIxMtz_L! zra*_UmK2R$+K&Gr#a1Y$@|Br74DK; zF#$f57R|f;X|rX~N_^NIseu0B$F)m(f?5d8!u==?YBoPA&<#b+5-mXaf5Ms{_k*7F zmgna`N@@GwjXs3+W!q(DR|9`OE1rFFmYEPzH#!q6ZNfCnGWdaKGDuaU9mglT%I#n5#FU~T53^7 zRO6poY)f;dp3NqMpy^z_{jq{#J67C!{61I>=~-Hx^j9sE;*=Y{qo|_Q(V3ojr-j~! z=~0#h@lvA&9M-L)3pb%kGFIScrfXto5$*X=j8XxkfTOR{Az`>tmYn8%Ym_w99!gHjUqRo|V)5 z9bMPjfnYg}dwbUz;)pLKXycR|FG9SGdml>ez74>Q9sG}-@s&&i>&m@mILpn)Ovf0>E>Tpsq6XzjJ~f<{e^** zp_C3jT;uT-V@;6-!GdFm<{KG_;ZN|l;Bok7s@sz6;2prDXQ!rrr7?Wz8w$$D(2y%= z?_()ywKuethD}%O8k#;g6>dIyD=M z<=t*Ua|>Hr#~ul08`xu^JqP9d>L3b551Ew=x#zgat0PsJB2dgF8RSTzhwE3Qywjwy zx18FPpUohMn<{Vt&eQd%(Q9*cDQx0mO*yteh}#*tVmQWoQ4gBgc$aLr`|2^D@ILis zWm$K9p5eY@l7;-XIsFYnmaisOLwOW=!#RYw9@*ph(a800c2Bg6vO^WtA+@&2Df_|l zp?~`S09t%;0LvkA%ew|rGFRIJtzUykNruaHlIhd}CU6l@f3^;I_sGRM>J-!DLL6P( z#MsOH$ew_D{ycqaNX2g5jO9ISwP#0l;7v|10@~{qalOn_J3F$j)j$au=a4w=1y{HW zaU{`BcEK%D!YWY13`GDq@Kuj?!>lpDmWyafQL>sNj*$Ikl+Cs6cf{F12Pe%9i5h8CxxM z1~!mJRf6Ch<%U7X&P`F0&+L8SC6a9{h^>O(T`0MG*-VTjz1s-n3Tx= zByvwv*BLbhf;f{&xU{jF%0yYV-YkA{+(?_B?h%dwRU@7`$Eysmo2%0%i#q1QFp86~ zFq|Osi6G@$`@KGJ0qIp@8ePoVRkZJMW~(7v`-VH!RKMO-10xI6WaRo#D_wtGgfPth z8@HLPBWPv0lr7@6WMj2*gl7N@f;j3=C$#Nrrrqk-r%Q#Tv4Yr19CO_&xNU%75(fM+ zJOu=S+pSl%vsrY#M)oa6-rU*57C0D@CD3=l*gc0Q`5L=6nR#)j+S=Uz0A`jNV3|rX zBKcw_ak#ECCOvs14Ca#1r_`$o=stFp_Je+iLw{)ORU~}G4=W^RW`wq~?XmvbzJS51NYcx4gi9<= zue6r{?#>2qdpXBBrfJ6H>JDrcE9v24(Iv!NgT|^!d^aC-bUEwA6HjLq_N6ths4kuWxw7Pr?B+G+O*XJ*T%%+f15g4YLlIC0g${RVyNc8>0QS+x6$J27x>W}0baAWIt} z{K=f-9dHLEFnbI5N~Ug4trAhg!J4W?eL zrM0#Efu}fjDD7t1dnI@Mey5Dz+DeeLwpR9Wi==Ci_OiF`p6V$Vkw#-19_fQxKs@y?dKiT5)J_Ts!`M< zFLwU`*Y#qYf=QtsbjZf#j0IXIQic!$XN+X$9ZyrATD5B};#^OArs?Z6K5P)QH=$=G zm=F{KInLI`20-ssR_$TZV`y$KQt~^qb%lT3M!XHfxP92?(EC-Dns?Mc&@ri&p5dbs zp)#M4Vb}wlftDPc6+evGGW9a0x_(#JJ_&M%X%qXk8*l~MQ0Vtf<1bA zR-N_b&E2(}S2<{HqA6@P0fa8hMnLE0X8cL2u|vLKy#pPm+OU?a>4u8zs_I7Hr3a?P z=bnAdY0sry#cF43c>*cfxUU(iuvpwohwSjYnBkuoRsR5i9tBS;ElDz)o_)JhNzGpS z4ZJP>mw21M=bgIwjrs7S^Q1R6(KanD6d(ENaD2x6zUcg^Te$DohThKNeE8Wk^ zqw2?slgYDy{{W<1H10p;ov`EabI0TjL3MF8%l`gXkS;J7@IHf}sB=bx)tA4t41e+E z1XjQ8kDgEa==A(CRh^8EURP&h)tKa(j8n5k#cK~cgl%O~d*F zR6)V|4Atv)@-*jv;%(>f#?kt_epON2iELMve>!KC^SRiu!0t^qdUoCVRA7BS8gK*J zoyyR)9o$bChUrV77*_)iAJF$6$B#^PryFw?CXKwvbB{4cPhvYC@-tPG0(*UGcrH!3 zEf*YMoMbTcBcUF~sY|(=k6|!OYtL*|>njxh07sip6Q!)BfOw)TPTx%MdBCeOSy^7_ z=I2jlOU087XN>M#WU>-->GJY>zPIH0B6;V>qSdT2)37*uJ0ymOHaWEl$ z17K%8Hsp1!?sD4i`uz(_XXe@~JKDO$ZQ{DT61$V;Ws_*mScMx(sN-fj=A@R|S>%XM zX*6c%bTECTgb-VA+am25;eKXdPXsVHs@GRmQb{!VYl&6Fp`^qR?Q!!W#G`^8w=M`6 z9XYEu_KN87X^QtU2aeWFLN#VSYdL71GTTBH3$SOfW5#K*`H-Nrcq5(*8)>&BcRNeV z$CgUGM0H~0_l>mhIu!tlwQA{aA|zK|XAs;*2iag*E(}tT<;qB1JN)*(_I6!x}ZE$+RjXc$K{E>dW`B z)aN+BIp_n@#4$o##b~z9l3TQv(^DNh;l1 zfF!rO@~z=5CB&+~>z5$lD;6xIu6;c#L{LQ(jd5cGyT|6T+qJ_fVB^dzK?i~|0qR$b zsm0i3oQRUvJBwrTvW?=>I5!~@#_ZgaA}&YFKJmag#Q@N&5f@SjXK8NaYltrHB!N-| zNk(F2+ycfJ?&pHI$f+!+fvzlBE`Kwf+q!MZ0F3Z6RAiRb?QP>Q zTX|B$Byhs_4=cc-%OtOkH;z{r#s+qbK#0vmC!=1`|X9BH2 z#bb(FUC_lSD`z3vzi5nPIhdc~0N@@yyHz%m&1oL?aXQN=Gs!W0-9I~SfZLs`+<*1) zd2HjET1%($+UY@g=3EakM<6I4C~!v7Mh`ytuj~5$zY_*)X^=;62kyjT8L_e1vet(< zDm&z4dm7w=JDX)!nn5WeDBJ-k>620DH#U~D#US(D^p z3^u#t$-&*6^V*_sTN0zxvTvVH@a^39ZXF|tk10>ig!JRmyp~}ZZY6L#Zm&Y|eyH}o z9av$KA2#Amq>+aTJ9~=ri6pp^Pn#L`6^YeN5=f~eNU}|hgN_e5sVyUd-sU-0Qsf>` zNB4(tdG0uC!PTOD)qcEqXwE)-p)3ssn1jB2c>7iL@@cS z_t<=Z_UEtZ#X1{ToEc;+5FBuMV~pT`GJ6le5R`c^06Orf7(91 zhDr22N@zD4#+tWr+8c{_!3dDt1XTm7rcjfEoS))3Jl82{B*()x0I=F-nEby;v-*EJ z^&bquB)$)`i82-Z;6VQXc!6BgZ7ZA7jAr>)^|9uc-(>#Jo(Md~wJ|i2q`!XJRU;rA zbU7Iu1IKY(ODbn2U9fuBVX3n2zsGcA`MiJ7H|1R9cE#jS0q5SfS{WyEGF_~GSlfY) zwGdIC^=hevl2@B zje_4>#haE!X#|QoH_E5x1Ft;PP@OW*?#ng7AM?#N1Nr3Ds3V?OgPF6onPgX0Kb2@n zZdtf)r+@`+$En&p!EN-o)xBkM`9DsTWBW$=tpt|-R0Dp}u|yXfdkU=)la!gArFeGc z@B=6FNl})6bbG2OmmFroWH|mgU%Irt)b)Eif#2kazW`%#f`o zxgzN{Q8Ib=kbS}atr>~^1u@{#p@he_;%&>Eg;f6l8cY@0oP3Y8{S7ujh(SqFS^zRwm+A^7w%>Mvs zTWS9Q16_Wd@dDa#5=jD|Nmm&BMR^j-flrQeT#r+s!E`;rJXZ*i1(G&a?ya0wrMHNK zC?SxK_px3B9os1wVm^koZMA75UzX-oIOq*DXDC?rZ|%X|qaRlke{RM906kzI@H+7y z_(*TZOUDD~DE|P2i%#(a$Sr|}F!g_TeHicLP-R#$w;*kAUJE5!c*;ULr> zgLfzAik{y7`g4;kk__?*JXA_i>dhlgTcg~Mh^+09gexNfo;j?+;< z^!x4U*y(g@uN!JJ`C3fcoywjG5;;NmlYvH~qb8Gc=lH`~hAEitYg>6d`Plk#!1|M1 zo~`jC#QLh3FEu&DXNHa}M5KCQX)N`a^?Mj}tw!28N`ft>7)e(=mEGKDrgP13D77wX zQAz$MevZ+o-GuRXi!@_R7t{KZsw)_k>?X#(|#kKQlg-|9aQ zDs47pxmL7x+Q;UJ$@{K4R6|(uGO8<0)sM`rwDcpK(J?5X1Ax2^dY~m3%GToHc<%B^T2~->djzO(E zdscYEUJ`^ZQUvBgI^g?zAHu3fELQGPz!#G^`$I3y{{R6W!~Fi0pDomjCem4)HgY=W z*F9@~be@`=+^$Qq{7or!IbI(o1tvfF=h%DWJk#R5mgj27<~dPzqvr!3-6zxQ)9KSV zIFWZ1*(dji>-F^d)YisH?gVb7kEl5`)SRBIITTii)wYaWT!{HuRN>Xm(!Yze3% zL#SIzrrN=ykqcTlGRz~|W|g`*Cy=e&?p%ciWi1~GkX z%WR%PGJ;H)UE7ZsQ}bgUqO4t&g*6qi)s^j-#-3OXzIt^S1b+#~Q}oS4bEr=5eom_L z8Li_H|5#~%mZbRUe>yk1vk=GTc9qsCBmQtpnVQqN=K@Gcz(8dvv zjm~zEc7ui_6Q5jFdn0xIn+ipLV;l`~_Wo4CcV*NABV=c8Iv?R6;{@_(QvSa_!q2HS zzN2|@J%rX1%WdVV+DOoV(kI_wLlUf{{G(_f@qtw4k&crDFLUR?6!Mb_CLS_E<=BzY zf`A53-8>HW?H^Z~^G!>2hftdC>1Bf1nPu}@Kb00m&Ts}mR_snVCprFJp&y54kViMz zB)kbUf=}JF!AV8OLO?I;!R_5#X$f&2l{ET`+d-%cWkdE5;Ym-NV9Sm%w|7qXrD*gy zE`W+V^@v#j0Z>{?t%LGS-&2awyiI-YHR*LL@eSR)DITD&ayQ&4DxpBmWaz9&$m4gV zDYw-|si4~Gw$~SkEYa+(EhJGRM&+FXl?~43Cy~3i98w+hzxA<|1X?&QF z2aGNmLvSI_eW#rF#WE=(O(OCUs9ZqLBxNl2os+N4mU0dOUNQAF^}f4JGQmyF{C{J( zk>GP7F-HFYDgD^&z>Ja)anrr2wF@~T*Tk1sH#1%olHTHYQ+%X8TYf_oVhKMh0l@^e zaX{|>0IyPi?O9u0ySkS}it0<70}qiTAOt2L49chg1Yoc@{LeI4TqTx?VSjUDcX1q7 zjkA2YwU^}^kghVVkKzZB+lfuryf1S}wk}lFtX^bj{{XTD#nQ^Mw-8-K>LF7h z$`;2ebDXdDk3(IL{3d$Ghx`$5r`z9L#J9}yM9|L7Zj~MxQR=IOLz0 zjyZ377Ml{{q}BY$Ak=jhj_6xl{Qm%Gj^*InhjSP#vVsTj@_P5rIN~XsDUU>Rn3a%KK7=*5bm|<3^B`^VN9~sRQOGJZH8!z!b7i zeR(=G^0uPd(iPw$duPs!fs* z?f_>wBRmt&8*^3Rj?tjf<<#`sp>ER7mlu|@=4+BU;ZF)zBkB*WS3z#Jc6T>=eax22 zEXgyi#ss7RfDS%VyM|9W$mvgj3mGIaYBv^F)^f%nNg^ssw-OEy$+sU;I|{4Zn^>K0 zbk&B_?RK}95u&=o6s@%yHdD&re+-1Tq4w`%P5!`3y+2XBNtuo$y0(ExNP?ZeDJPaY zOCEr3!1jhMKVLs%j@^QbfizZbWl^94KP!dff99IoU!7r-j62HBKjyl?&P2Rd5`5)t%fGwjEF|;dY*&zBBazw zTcZg%@&}inQaYc{HEUxxtdc7XccmPj1vC!y;78#~F0J=Y&F;A^pb|I^p*1`zj zk}HX(j^<6P7|8NnbIE4(`c~>mWekYggwLj>!Ge9dWP!(&RguBukmMZTo-%mkW0Hzi zRoFgw$irFhm!lkIrC@&UDbi;plW$HK>OLX3dBh5;n-qaBTj zay)NDQ3~h zx)giyh=g&1>Q|hz{G=X)@I3=%Mw&&2c%&he#?ngOWU=E8?Xo^{34fWg3ZGHMS#>*T zqs$Q6sF7Ad6j0;`Pn2b!BxDlA0zo*)r_6y}>|cSDtZ~6~t_g+|Sg$;s#~@~rGQ$H5 za9e<<*EtBP1lE#VM{2JwB#Z5Hd5BgeBMt)-lYjsq4w&H8DfLmdOL*pv;Z`{<QdrQW* zf*Eb)k2c{h9K~)GgoXCunDdN!anp*uZm%FQNG_h$L)&EGqgb7?gffs0B+2KD*Cb9j z1dQ-`?N_XBq~_i36(eOUnE@)i=eIpYIL(e!w7NP`9O|*jaH|!{OCa**g@{?ALA*9{ z4)y9gkULe&Xy3_ME!uA;6x(faCTC!c~`XRL;pu+a~6Wh1y$_ zkUnKThn|?L_R8jSZr0Kn!Njnk5i*c)@y16DyE(=?^sVCbIOeQewRJFo35whpfM1!G zTXZ=Ls(+7hJBstcIITNl=G;g_Z)f6Y@Srp8!xdrjfCnmmqN&?RB*td*K*DZZm2-{^ zoz$q{F!ZbTZ!ekysZE2-+Q1VV4tIx(8AYsab)7?*|M^RgrkXr~u zlVU|k^C{?LPR=vmjz_gbi8$|w7qYeWy_}Os3@Mh3sQF3s$5C8pE5DhE`ijuiF66z{ zC!J<8qd5rAf^a%j9rAAKTJ>ypyzYs38Mc1S!z~L zf7v8w@;Lm>B-XoiIU-+`9}cAd0CbG!@*=hTFK--vAhwrQgeIO4i-wpLQyUB)(D9D= z$2D)nTDG6(>G61S-O?*iu_fH+GE23$V;ukmzPt~mE9+scuFSbDW3*(GP~5*L^6i1c zk&ZL{(cjed@0vlV+)7A~5T@OsP+%(c$J0NbdZ#Xg}D=nqk?5VW&1azX+=JSV^+>k~{>F-n|tZ7FR8Kh!3X#p6j@x_a$^8v<9RZ!}I zgoiG~06LSJm+dphzY}MX)1ShaY&r;3Fk)EX5-=%R)=wrzjT;O`IHfl?8AMhCDgoox zq%zzac|Uq^st6eO!JsZjJWsk!z>k>}WKtM_wNkrEILjl9t9#?O)K%#1E^n>`Ra`7VUO{zTd;#fvuNeB(dW5F*nnL;uoA3yf!{{Z||E@V}tc>e&{nMWH+o`mE5yKw5y9)!dyi0O&?!HE9=zKs!f(UVbLT*(+K`?(eI_fAeLU1B~>!iFbf1|LIN&Y2af zrazivLU@)V8TzmP09uy63tbI|rQ*JW?KSCO@_dV|pZb#OqXBHVDmlnv;8K9yQ)bupGu z5bzjdf%W#PB*eZ9g;@Nhepal@vmB7)cqDYKJLUPvUy+IB8>irF8tNhe+@*c{R)|RE zwJr%2j_=Pky?+ro2a#BFL{*O!&sOK5MXM2RG~{ki(xWOzN9;y{!HVKD8+ORXYnsyJni-McVvMICgI>3x zY9HFx2@bDs=J9~)K5FKUA<=x2)qW$L9&KV%)6hj8LZKx3R+YDeRDSv9 zXsvU>ae`}d+Sy}YD@}`TKjotg*6pXNF;TYX$`(;+!rm!9(IQN6SrtC*WL?||FQjC4 z+FOEZ8zJV%OpZE($jvixd0Ua!=~Xo4G}+rts8f_u*8c$JODiT-w`H;18bfmFu!0nT z3YFramm!8d2U=^l3!h4EPRit{tFvlL$?g~XM&4bQU~ITtWJ*0q=yQ&Kl~Pe*s5kD^ zSlpkzAN;d#r}uwRRc$W3iGghFZXA14wHsivZy$H4BfV)Sr#|bcoFmP6Y}Vdu{s7sm zrL|zQMpT|uo`>jr)YeKwoC;;0r#-wT34+|)H&~<@1Kf89+*Qk4bh?nfn(X9q)~Oaq|Q?rkun!1;%2Et@fEV$?)wyW&PKiu<955PqkK+i^y7O3C}$8O-7GIX=^*O-Oa(94+&()-ixk2 zoOJ$GYAJW=?4kzyQm~dLVd=pHdxM|D-m~GpmL?4hW!ZXQe?MQwyV(3;rvCuK9yIf$ z+1u(;TzRry7{O&wI0FMcxyPyFn&_sXZpR&IRgAP(IK8a)Q$pO51juvvF#NxftkUeZ zZWxWq!``|GucVhzFx(`5Xm)6%5rqeLRpaBl@eCe+MJVZ z5?S2&pe&Zww|roH#TZtfpA;!7MB1L3CP2PeK4QqE^OMIPonqTt+q0sTe=y?;21)u? zL#0{$r&LHZt#V6u*zBg|L}5w*z!F;k6WrHq8*#h-zpqoztrVa2{{RJk1#LoeEO$vg z^}NDGM-oG5246YGN`jmg$s8PGp0t;;-P}QYJeO9NqV9dIF;o(gfE~aQxk1Uv7|91Y zrr%s#y~Nhi-dV{l$t^6NS||*^?c;!P*z@?)B}ryRyq`;e#7;|FzR(Pb`>=3$!BW6* zPd|gX5mrlY>fTGL{<>P88SV9Jvwz{p^JxGR+sC;CKfBnjc{ur*U)>c#cef!d!uXeryh);B!v9 zk5Z8xZlX)pI&^HZ8GOTU04ui`CutlmJ8tPgO3Q!i`ngsMTM1*++HFNGWu8es)gVW2 zvncz&C|+~VAfKSeHT=7AU}R^8NgTR)Ex0^j07RT&PjCiLe2$f@vD~J(e#ha(5<_df z>}N)h1u)>qM+d&~o|zqLnoDyIlcu@TVe*CJ$qn4G{{Y9q1$Rcr_>^FA+i~@zRrUV7 z$ft`*jkN`X?Q_Nu8!2`L?lNT@eU-8O-N_7o=s~jzdFls0E^6MNZ>h(r-A$mwabq=w znR|tnTuB=32_ffc!7P4UWCPzBq%CY`2AX>fI@kM6C6CW>9i)yxwk%W@V!$X3#~}Xz z3C=3*>D6VA#1|6BWi(4HQQq7`93UfZJA9%5z$6@$;P>rNn2o)zif{Bm_GfpQWkSgJ z>Az`G**yxC8$iZSQ&kFC8%cM4pj`QgBLkFG=XOtAkCy}7^TrA(YyEkfNe-`P;tBr% z;!6xST6Kr-?b_Tr#)2@OYa?V5cLq7&md-06POYUuXl)~frMfQnjLb_61qxIXz{?zk z&py>DhfH?VWQzK7k*ND8n+G>PFa$Oj>>*BjjxmwNUNGvyZx89#+MCMEt1YTUKbS3= zN5Q~CcAgZGfPFqo$f9S)4;t^7eea%u)a z9ItZEeu2!#%FJry7Rfx0ynAy|`N=q0jof~0^FZQ)9D@Yas@x-HQAvxNf?98yiIi^e~0 zwu#71s`02s&Nw8Fr_`M1^{iaF9WhSVwaSeoa`Nl3zr3H+R81KikDPj~H!6PiN79vY z8z|iUaBDryt!;}?-U9%(yaro*{G@&8&#Is4(EV^jCA4c`Bv$in+~H+W!9Iu3`+qUb zN0BCXFFo0KKZvJ6Jl1=fBC^865+e+&4?;iBs5z>=_ZHg~XsdF;4C-aD&j6RVW0p@+ z5AhGwcNiI}qNS@8ox+^F9Qt!fihoG{r}U(Om$dMy<-3gh?K2NmL6P~KkHV<2XKc4D z$0XJH7s<7_{{Ut<;2%@-H|h`PQ~=2&cH|1t=&WZQUBixjxD;cKqw}VYl(?uYcWQE; zzmM{!1e^X4B;^yyaNad6tmNfxXeH3vyU5CFmB8SRl@#aDhElUuz10EAviwB0&nR5tTOU5`R?dE+1+rnw7-@=+y+M8 zG3muoYgr&&?`Xj>fca)$%|WU;jPz}%{jPsnd}4TynFBK8_@i&?DRW%48xPDyxR5Wh zD;$1xA(3(a03SqsK_AMK?e`9GF#8ds%#73j0MVrX09gvcNybt}1M)ehlLzKX zrxxSPzad!UDGkqT`c=IyQE#Y2EH@U@OS=k^!?{~LbCHhy>Z`$RE9D}O9RP~sXdm~| znvy7^w~=H>u2*RXcMiXfF;yGf%2sN}NoKN0lSOf1WU@yAZObr5#`SCwyXNCOb-_NE zt$3of-XNW0v|li=-*RS#N67(*Uz-~;HxA$)=CeFGr9P!}s%5m3&xUAG%6-P$^5o#{ z^ye7s(1LY7!>Q>P?+mu;mk^;PqfDD&yXImt6yzQNBoN-Fx+AuS4NF{}Ek5Do0@Y!- zaGq+$t>sMumdc)So=868wJqbh)1<}SthT!s8Emqa2RIGXbMp)nhCY>%_KRkcNG_vB zV8EMcokr7vmy%)s0H6cGC#7uIUkIdG6iWg(%sjF}XMjLl669x$@G>%L^*P$^Y+K1B zrIB4o#vF#+r=A><(C*;-l0|M?X!?$$V=vg7%D8276P)CO)N#Qfx)D??FQSkBJ*<`P zI9UoXUiK*J% zNge%!u{yZ~%$UzTa%v{icCa!{YY503$PPd6jG9eFn*RFY`ZFYkMlURK5@d{?x%D3Z z0G=uYmK8!jr5`4KZ5?F0C>_oXdhW^xwlLFF#stg&Qp zKic-HURscmGnIJ?pIig~0M->hkg{AlK`CJ~tSOO?lyX@9RJH?eLGb|6t?#3@vbT+7 zvzvQJ%_9}ykWN771E8$G?K|7{3jl`Kc7JsQYY+f+EZEL5o=K)!PbAV_2$fx@h{GD- ze5;Z9a5|rQr8L4clG543N0H}BKjpIy-|hp`)Qq0MD-B$I)yztP6q(fL zVg5K0Pk55ZpWh7%5 zLDqz`8$WvdfszL_{{Sv1$SosdJ(hx7K%z7sJC)<2azOOWJq#yz?v6;~Ez+8gXp9R- zG=tDEU)P%N;_&nMZ@{bKh~kFs>P1&V$8uV$a2o`C&(L##3Ga%TNkq6)l+2qi3~CpV zrNdi`OLYp;PjbQ}7#^pS`BlpsT@D$VD{*z=8^#4qy^Ll!LGQXl+dcYK&kfmHYuXLZ ziacF3w>H+>g2e7fF*Ly_>+L5ni;{}-hrm%8yzmd|Z z#tA!aXbX)|^KBfq`W3kQ*4kPLrN^U4cn925?er@hEzQi)M`tVYZZBWYxqooIkG(Qo zRNJ4kX^6IHqa|Yiv_t#M$N10b+NiwIf?b)KW+3F9#l3P*dZ@wMxApyZGL>YsMx*MG zTgXj?nRy$Axw&k!eic@0`?-(rZdpP87>3%SB`?4j6!c$SwW^9oJDjD|e$`|1ZgcAF zO)ruUL;|&R{{Ri&>K4Cfw-FR43K{TQALqU+q_prld1&{SP6=L8H3J9hQ!0^;?6^>z z_D2pN$FJv33hI7Wb^0-{L0<)F;yg(#%-`O~6`QK~V%E=V$15rje*XX@AL=P8u#>TJ z!&HL1IaIs7i~U{0pJwOz)SuedOuH_vVz`X;1a+s)VRam9D1p4DI94Mc{;FWT`z#@1 zwPQFxefS@hQ%j;VjGB786eiJh8^-gZhT}-M&P32~0s3_QRUwnbx^x62w*Dp4$M8)U zX2<)dpU9e?T}JBQDz?`rq6onN{#?}7mVzK1=aRzVzU+f43G^eSWhkxLv?@E8)_Tu} zZ!dP;X-#2woJg=-n49PpQMa(;aIJ`R+Y@~pwik@y{z7eolBjzX^&KHN4l(Dn5d8s5n0 zN-bad{=Wk^QPU>VFJ!m4jnJt(cd1g#>x0^xre$tnjBrK3Du3P>{*~Kb=$gKZp{@Om z?0V*ufJpY2v4YB{x|7F$Z^pUb5y@iq+J32|*}b4Z z?T1Wxsq~;OVo$x9(bPy{;Hm46YN0GhS127+_Z;B+)}w$gqM3$Pbi;jmRY+50)4)bU zf;sGIutpWmp3Yog7Xxs>`@*rVbluo0IB}J(y;GQ^Jd|y%^r$D(hnGAmIf=Z)`gf^% zn#N5W1lDB<1#m}NuGZvsuFp`@gD;+|#z1lCaanR$ZCv89ln$uXIp4FK)ws0s${Rcp zTQTVmDzgN@QInrad=?{=U~+mM!n3h{(C3=pX~CzY89W-*lTRqcSaJcAzqnD;eI!MoQ+MgQCrMa~i`NIb(pTE7~+&65GSpk_qg8uWbyd6{A}_}q-6_1( z$?N|?eJvbzvtxE*Y1j{01GMtL&r)PM*nawuSQjX|r#>mAOV1vy~7+)_f#@<04 zb6OgVD>O-YZ}(sC4@2o$$VXS=W!rLsz4#hJqSDz zkLg`8clVjaQ8b)mu?5*HGv}ys2sFuC&bL-Pw0K^fyM9MMn62sT>~3_}=T=*1nToHN z+mN#3BRyBB>ygh<(y_kFJTn+&j4H~d0|AhGf)9Q`tXj3?jbkokqW38IZsUM?#$#{{RzJDB0<;F`d@Ckp~Cw zJ0D?IZIVT7s_cyLN#i5&sYQgyLB+kWAH^f{N+Ge&`c2f=4e1JhQ$?b;ChhzpvCMJ4 z^T)_rJv)Q`eJjztC*gQJSt*j?KO_H$WWKrC7|*a$OC zmOl6c8LwIJYelJiIJVPcF9kR&eA`+^nYaZV(BivVQIH|_+Jlgdo29$f4y8i&f zO%6RSd+!ZvaB5P-HoSjk+UhaXXRqb&UO#q%1XLybxv!(aZ!` zMTaLKp1nE82EKangks;s6R12l&BbuZ@>E=}NBV!mo`oCE4NYpluI$W$Rw1Nvr1AHL zd)C86J6f!4ZyepCJ+c^sdHYzR4i0!30DcvMFtEOGa0V+=M-tpm3&VR2q|nO}#f+<# zEPip1PNb9RU3BH8yO>prT9&T@O9lPHNA`tjQa?KGP$!A;lZ6fQ1Im&?=sFWgcVz@u zH;Xl`#+Nm;Dh0g1Gd#R!ImQ7QY;%kctw%IU+DV#M)EZB=FbvTC=3v)ah~ATt=z#Ye_o&09`wAk^5~ zXje|=5K`82yo50w(hrn$&hLFLZ3%MrC6;m30xfD03tK0LPY39%sroZUh1Mf2CTI-r_w| zP_q`AhMyPMfSTFROCl<&xZRMrQo&Ae2n2QRpJ3bi^ce+}elh(Sw!;f8O^X9<;GRr?Xf| zaV_njxQ0i#o5{%CmI@F8GN%d&&jh!nJ`FnJXtXOT}wP3Q*uQj=DBDB6u zCL)kPC#Cqx{RLjSPe>hYunp9xGpX=E03_eQxs|L3y{UOHc0uEeqr48 z>rI@khlfY*rnH|>ZKnq@$!>%V*gWN>MkfaukEKVLKFFMMN=7ejpCG1yr0 zNnY6m^vFJ((AIa6X;^z`XMn)hMnEvpsgOonInUlq0nc24%{EoDTR7mjhDfIn`Kus7 z^A=%@Qn1Nj7z|IGJjPZ)IX4bo8)`DL{MP`tMuN^K#42Ox&Wd=Z}6tjS=SYe;QcNiS{USB?f-V!KWRvf-N_ zboI|b#POOfZH0T6UdrObnG;MYCz zXT!Q~i{b~iv(zl0mOY@`oxzk5oR9$nInTaoWcqa-Jzl4nySr{@Y;NV4_x}JIiy+7_ zFb6?a+QdooHlt=w{d6PxQ$Mr0`ERU`{kH=REz~xvR4F3R1vnYyxmgz;k*ZlMYF-lP5SH>J{u0U$)4%Cj z(CJrF!m*nhr-T!T!tXKrqklp$e!Pyq!@mi2jaNsx)SxQLW7}<&AeUklw~ni|a(mO{ zX^|?iw&$MR%39tfKYB?XcKl8U{)76}HtW|FxvThlP_Vv|JIIpVQ{)XRfG{`**R^K| zwD^zuMa+}8_ifi`{{X(5{3;}!^)3!J(5JYM@ik@(OS_fb3@pk0{B2SAo+@v)!x#K{ z*}(z-0G^V!n*RW=1Mn3D)6H_(o>?Y8?Pmk`X1K;&1e;#bzB+-8{qp0DT?> zOZKZ+f8*N8H+GWc{MP>fzIb2KsRWZ8BLVCwL{grJ-<2|2?r-S6BGxT#-|X6r#42Ht zLPFkJew4Qo31kZpA~YLN9>di8o=Kz+T?qFR z)9|92SPq|w^`?`Wo0nnP>?!LZQWz()~+Es zewA&dTU*8=NTOZLz<(3B@i9H?ollfx{sOF9T1|Pm(`rVaepwFGPpSTcn$bz6V;Mp> z=3+FE>=avZ2bjU z)6uWBFF#S8osoBv;JX|V+ZZ_OjxqtOj;x98bGh)BjNd|)WPxOw7b>pTEM#eLRPccO zpDuaH9{g6Wqp4qbj_k>E_I9(HO~NaQ8(;<41G8@E@xVAfnLO4C&nS*pK+5PO{mgMk z5Jo}+Mx#0VuIB^*0YU!00&;qgYod)1J?xh zs=kIc)VXti{hehs%kGFn#BgrL8~6YRoOkWVL0m+dwdD6^?%~;{W0N5n4fX#3Ju2p* z;!C|q%^lK1HKszwl1H6`I8%|x>yyuF&q=J9Z-Pl|m>xHN@du|P<{wNNlW4}ojT}0q z!pUa;0A{^RZ9v>P{{UsbOdrCckt3E?is1#l%AdV}5&ock)9yI-?@WtNp6QvRjNEM* zGFqMNJ=sSfeMLsA>z z%0JpMkItGU?{a@3r_|mwAxk+Mk3|C{vGxEH_*3L`xwry7$pmYJ2dGigKIf8s%|9Vv zna>S!+jHp{Y@f{Feg>mdZcr&{X6UJsNzY7D5F~}!$W8J{a(w`)(Sow3OU&I77bKrb z0FEfsTu4xX%u#aZ>Uu99Eb3?MSwbIrZVO3_3s(`0&DxeOZk80t=1RGEQDOJ8< zlS`Fe8Px!}spD6s^ zXU&iL_#e|0&kL|m21H}V$mcRT{{VQXIM3sXe}EKg-w5@+9wWZuEzF7El?=c5@|w!{ zZ+@4ycbX)xwrQg#AW!j(arlAy8p<~1ey2p!mn-tIa5H&nAMmLp{{Xb!r_|Gcz@j<4 zrfBAYBq~V55(qy~X-&jurJ|4fXy4;qJr6!hQR~H7g-bfFRAhmcI0Nyjk+&lw=ml(O zwvBmkPnjGs#xn9S$sa-7(rqEdHtcm8R)c+IaU7QS8#j-hKr*=WE8PD8oYzATmOC~3 zZ18=fovsrbeuKHLV%J4>w#;{uk2Ieu+{Qyo+b8DvN4cwTXjjv!hd~lY^01KQl#fIF zL-MX^B`w+C2PmsX5AQeDOQ@mTt@g3N0ItUOK8N(IZDUoE<`=gbWRH%oB4jTa;PlOF z*jdYM2bFNkAXfd^9fe&O>57QjF7~<7>CSMq?f9N^c{Zf$5NDDl^2i5oE(y;*wUW}n z11{Mlf8s8Bn)TL(DOVCAu=g3RC&ZR^_PUuC?`n?92#!DoKNH1m3Xzkwjw+NVPgd@8 zFLGH$z(V?W-Nj(;FCR^|I+V}0}yrz7Q=e=nA+Y_g2yrIL5_ zMGC)}&+@DEFn)jK_!6=U{W5D^+nyydz{98`k_WLBscfI{l=uecPnYJnf4oaV@%0lj!aG74_>cJ*@6Qz?Rk~Cj{r@ z9)hUCu-jZj;hako24HyH4_?02x2x&578ce~-pf6i@<*5Q2~e+WX4BaDuO zH-2c66`DDMVT6D%D!}1~TBOBepT3003!&?8FCRlxr};0IPR)Z`m#`PfVS_4dZ2F4H znj+Tm9CE)Y6{X48uOfMO6DBxepX*ixMW%LMSyf-^q+pNMno!E?fJ44S_RM{coC zE2^-I44-+pKA%ifHd-kCfhCYd#>=<{-)0~Eaf*jY($Z0N46z29CES6Banx4C2^_Zj zR51`fRGtpT_U(gHK&5V?zg78_%Fdv5J|se|=BRzSE7aqtr@?O; zzwc9U_N{502>h57h{sdis=t?>MJE(eld$0hx~OG|zdD;Sg&P-bSdA5|8$GIok_Aj= zlwzKNcYB@x0K<)6$<+SN?ca}aUH+t>VDUwz&OnamkH@8a^J}eK>DSTCY0INzdLLt6 z#bKy;kHtP8j`5_l`!qOX3^DGIkUdXB^sfmotJQSd(e3)4dRD67v@-r0+&UneFrnizgGQj&*-kmftY7m={7z`SxenFV?#(Gvb zr0CR&TQktYKW`axD>OGuT_D8pv;U+snw?sJL z{(M$N{j=N5(XhtTinRHT%NgoLU5F;9V%xL!n1?z3CH4F%SC2JGNp8OsRS9xJDqBnN zFyaIbMPHNzwSqCmL6L{of&7J45i-r?=XZ>CPj<40gcB^rYR=4qG-`g(bSs znkCD5GH?2B03Xx}i$%G=veY8m1e4qOgL*pU2R(V|^{$7&Ggv0IBwENwwEKt}Ii->C z6Fi0kxdaNQ;){JdPSjr6HeE8DhuiY7>Igl?IjudEqa`ca=*m^2E7P^?Z1}rRw4N9y zw~o%*0EO0I%nX6K=OdC1N7Ppq>17G|rFs1VE(hjosnYd`H9aWlx9I|d@++1908(*) z<0k+f0sI(X=RBqTsI$6~8AxKSjPA{2IJipdM9pZvcyhha=&$2~U~O#Rk0%CM7tCSr z&lPrUN=-uFddSHvYx~7O6hFTotz{And2RsU?Hy`3=JI3?r)ge*P@gu49C=!k+(wue z$Q<{`q%yM%Z*QXUQr$-j5jqeN`j76PN`pE3xg*YdUm#ZlU4rc6lJOxMi9?^f;>F<5Ah@dXd6q-h53($z!Mn4Xfaa9 zuWchsS8`B^5>GMwr+gFgg1;&E1J{n2>W)ciD}P^sQb@G_0J7Ye#v~C!ZJ16-ErI5# z1$pN<?%4l>v(GBLHo|>BevfnS}Qd>QToel9h(u zH{WdoQVs@G`FsXb(OoEyB50mqipv^? zWy!}S^T$!g-X^p&a`y4bFx8-h!j|#R8~K1m_JTkIjEt@T$jgFr+N<8r_Pa@Eyo%=h z%@bUFvH2@&Jd%OC3NS+|_Qza*4Gk3*?lL_$P_vmN3wL9A31uw-F49PuX?X-~+Zzo3 z0QKuPQjPj#!VIBR#){qFTh(9}>7}+O5Woq~1NO zra3Ot$ptteIv>xc;Z<%gZoEeli^*Q+QMW*`$mcCIWB3W~!?$uWJ1DBUoT_A4qR2%9 zgWJ-mq>U7c@v&D^m1Q^}`qwu$zDHt)ms^a_r6H|L$YazkUsAOJWl}!Z0*q}Zk1>DQ zC#dXCp#WtbwJlh<%b5i(Ii#gtw2A@&$@lc608;+|593Kl2LLI4l+jlrw@9N|?{cWa zDH5DKzW&Sa`De8xOLiY+Zt{a9vUxV>etAzp><955Z{#vZq4-1Mo}uB$qtW!0jynlA zpz{V~X&hl>VsZu!I2>Selg)X(#oU)nySAih#BrzHLGAuMPkM|Wl#jx+Qn8ew%(XVP zJ!UO>el<<&s6}a@f=Le|M*5Z}mvi@b{Oa77QCtD` z>t$ze`ELdh&(wX^{Il;&`xD5etl#Y_IUz$tK_D<&d4KZKKnwd0pnp^NdQ~<&VS0BywPJQM zGZr}{*HEEKV@C+!R` z?nnHxiLnxWSbguH{OTK<$?q9N?eZ_&i3Ze9pcPPXFgU4^LSVSbJr<}|>dB&BLF!LH zE2h)*mV!kywWw@Jk}UEC+Q6s;Fzj~@`0Lx6%h1cc#Qs)1(-zdoO`~%KIW9m0X(RmP zZ^9JZeD_i$cML-)x{NBLk+*)pdm#7=x|$68MIEh@z&-WCYt+FUBjxyV5I z+q-rgfslQe8rjoPqaZYD@RUS2Y$*V4Q@JC~S0PEsBe)`{q(H1~_O{yFUwF7YG?0+e z5L6z&dmRYHXxPBAq;DI>4#32P18)BSyu{?6;XIB=VhG~4I3&_G7^J#T4Ixz9w65*L zX*-9?&(Ed~J7Cs6wW+YPTN`P2CIOiX9itrV=dkPbu8P{q?)OhH*~TWg3c?l0P(4DD zer`Dc^sX9PN3xNkzG&fa+{RmTeFzyK{c%d`V>m|c<#(8f5g&rZo1p2?eqS{WMFmtYH!%+)QK&j zx08IN$0#79nK0o00J^z9sI3*%rK+0oTnOS2nD0@*Mox85!rP#(US3yR%!q-NVdaiZT2mwec&QjE141);@rg> z05HZg&#pP7)3|gsH7@~QT3Pw=%@38hjkCX$@$gR94bbpMUZSwB^h?yaJ0V#>R2!j@ zWCZOQs>-_3Dq{0BzB&(t1`+w&ENEs%o@UwloUEbsa97dkZT`XK-?H&GyJ?{{s@$};_YUaM&4*k7osZPgcUuM1+@W+W?dCsdkOmwj zv_CVdj)9x*uc{YPRoTD>KbvBPWv9@)*zL+lT2; zh^-|1w`M;p0D&Pt%9yT*t(99SPDa)|(XQ&MVr#fGDHQ^kHCI%WCB`^`CgqTGGs5&0V@l2>hG%WYA<>y`fiU)Pc7E8-hX zUhvy$NOWy&c$(JjOlu+a1oS7;rMcH zo`?=jRB6U8=!w>x)uhWNYkc5YWcuzRtU-8#0c>GnI0cIkJ^I#l!=;R2^%@Pe@<&AU1vsT?0#<&#qPdLD!#;jdG# z_MIy5PJ5DO*ouVwNBRO6C z5)*3uGI>9RQn{Z?f<21#nFDtGe-Dn7a%I8ai3qClkH0uhb#>25rar_L0k@db*ru?oEk-TwdETR<3$HDtjHHpq6qAFT4!+fyCx;}OCBWXw4iRJr z@~=+2(%)fa00G?K3g_;0{E*`)yN^9Lf1g@@(#Twz*z?)!;DyRb9IC^p#Z+kRLH__) z!27Lwm9C!#qi!O(jU*ryIUhIYUR|r(eV0B&u#ILaH5y~I&M@u}UB z>0IUY)*OIqO6~9z=NpDTmCou1=E?mlWRgOPypYj!=3&oz_5T2cJ}4HmeUexC8yN)j z_pcB03bzb?gjTMXt2M2}az!&iBAkcoUKTc#r!_ckk8*^Uv*eb?)7PkcH>rPVS*d+G z?2oiYyo4wCg<)%LH-x7C-O+)YOt`_8HRy!&{6GCw%jlmJuI#kK2CZPam4gkYM1BT) z5;KpXt!*2}dL6pUY#Ht}n~{kvXHZ%k@W397E2f;}_f|P7MOxp|K6Zg5FC2^sS> zZPR+4+|%K>aM>puXB91?$9Fn;qmsiJKhCwqN-~^ZM2+yVlBUw?%i2Qun#8S)013z8 zT>a*L*8c!n>Gan~?kABK07mDZOk%mI=0kAY4!y-qd2>{PT@k(-Jh*C&^_S#`Ksc)w zvAfKn6t_KvBFsSfT}QALa>iz9zt*yxaKTUXt!XcK;;S!oTe8avCf3e+Vv^2zEpC!T z1UjF(!vOmqtv6DhNp1j_-^OH5oqGx3k!%OP_jyk->EtG z^yme7zlf4pwRsh$@}H^6`B-{`+O|Bn?mdfk zQUsN)t{5(O4Uf8eXWQ5FsITIYn5(H0#&(^@l|Gef?ljb`#oQ!$@&5p3Sw0s#)J5}a7Q1Z>zt{_nOmwII-IWk9_AmM17b#I>_%zJ3e4xq-N4S=b^U2C zVA&vb=-uiCW*yE&OkLKdlaDLVxoxv~h1Ho~j(dC3OBsezjHjmrcluS$A+6=PEp^FN z;2!3r)~?LbuGEcAFuM>AD@xE%<&<9LbY|w|7Pd3eKQV%{VTVOgkSfK?Fvd9J10PC; zGyBd!8wN=LWS{>4T@qX(OgD|-agEripGfx{7V;vHd7Doe9YsXJkIx=P!^011nJy7R ztMZ|@ecb(jN|N_e-V9dAu*6RRPyz?0deO?Fn@k*}+Ex>4YRXFCFZ-`SKhyoc{R`{& zvfWCzAc74%oC11#R+<>~n1phbj{FQH%y1|6bI>=Vckk*etHtGo zunpw6%S=hi;!j*|AnnIPgWsB((sL!OWXjOneVALgjU#g9fhrkS8Tp3<{KKi|6(|Jx z4RG54^I<-9kSZ|{2bJWWKBI=tNv}>RTSLi|rOfUtY3%Y%w%<;i$nb0s%tp=Qa8=}S zkCn5W5sH}Uq0}GjYoLzT1zIbB*|~F7K%~hY9Z;RQ z86*Ht`^Tj|D@Y`oJ9ZGmBj6Rdwvs)J(g^SmJpkN3q*bWuWZ(6#`I1FvV9+y7ZEXt& z$TMD)lMFny%8dQj!yfxjrD(-tqGT1Eq_Jj2R6!yQym@6$EtyV6bMg__r7n!J7=eO! zE3^=>LX$5H=g~Hfpo5+oy~n~H3QOArywoI+$1zFKT^G&-KJXd&B#xg<4%A7;3hc`B zl$0a6Z zh4HhMgP@iPS7gx>!>{xpfI;j)HA(bjwXZTM(}#bhTE4RTGo!@DTWLnyJ(N4>`6yinx z@)3p~h$fQe+}hpC0Xu>=6~DYYAH-n(HG85qb{Y;sjP|8n+t+68*#UO-&$qn+8QoO@ zW;_5p*2a&e-Rd`MWu#llcX=j3F%E;i4?sOw{{SwcR}*O>M3w-eEnX$Mm-w0^i6ize z(4TH``c?fa!(ZEXG7TOXKFmKjp^Bu?ipd z1x;;xN*;t8hOm z>>$?Xv$bn2QV8`eG1xO(PoI?k0C@=I8=j-DeARtdOw#T2k*fH987|^35#ybKHjH`( zP(Ez`01I?DJY-^`P_eXUbDB$MnlU8KA%pwv(ESHz#RmOq_YI()GM z_NgGW9sKEX-}lPo{*;PxNUEwM>C1Zd6lXraTvdRYYJaS{xwyw@VG*hS0D&>j=~G2L zqOkitgnWM$#HK<&0fG8eBtvAnJYH?X!V`a)85@fb7aulGI*vM5c?OBA%CNi;U2^TS z%86hZ3dIWZ##pcn!#plI;RrDvcMDY>|RYiPT_cDoIjE86%!~xi$0? ztF_(N+FgRlG1dI4haGs}WA6{v&lc+)FA|JJX5|8<9?myb~9s2doTz7de1tU2*{{Zz;5cgSr+aXgL zu1VZT%_}Rlx+=KK`qMlpq5UGe$$necEF^8p5qj* z(#3BhXAd)NCmlfK{{Tunzb@54F}alPZpQ@i{HZSEaU!Az9$(9p?tA{ceiRD#W38r{ zCET7|$fG+@?%F&31zr)`v@*>S%RQ@s=R(7nh~p&o>%)CP?hZsY`wMO|s}X?3K>oFM zC2LD{nI;OtQl#UYZ4{8$h_7iNXxy|p$~G~(9;2}t9DhO$UbNk@RsK=2nB;Wjhw2np zQQ&F(GdsY#>g|$3!O4>mu>gQr0hQwekF8+ey}TO2#j&G_LX7UtSNqxSIO;mo^Ih1d zA9m%<2Su~G)GWrkERb5rYDYcCVn? zHLjO(V1h!-%d|Hn08c+bR9urF0K^XUA9Q+sDiYtjT$S(6&rzDa_YIzfOUss$<8$&` zl5vWp@*V7_*A+G1KvSN1AC**@zkYr2e@bOVXlCC|-(}$Qim4ULN2a`pM(Epc$OjoX z=}*0E!bjG!Jo1*(Mg8?3q0MC}S)EXY*UXvaTZ6c;LQesJ!Td)`(6!ffEjrv=$+qUp z{(Y`gO$qt~_kOBD_pB2q3+zokIo>8FHY#vBAEk5iHL5zC6VaVCb9i4;Vdt&Zi9UY) zzxqrg{sP20{{Y`@^sbWF_>x;yirV8$@s^x}D+J6Xkq5HA*_8hPyIynlh~(Jta~ zxi~-JPmU%8J*42UIUMaH@v2fYoE@}(ey62b_@_v;-(s_Bo)NfXiESo~-dYp>)NVJA z)~`jU+&B7dj*VxjfC2kwhi{%a2f2iO)9tdn{$Ce(far2*m+~8%SqR(qq!JkLe}Ru9 z@TxXCquRQ>*6_}&lBmgy`x3Z6tvjyddpTP6{eQ`x!Dp%J7ZQ0U*HV?&hD}N+n;zSm z5JhR`RUbZyeV*nEm>7OuRe2%TG;5X*b*AcZNCq=$seGf@2@%`>03NF=toU>e6U3|R zFQGsCMujFzrPTF^iXXkEc!^K@Fhc(T$pty&fq&)4#F_WCjequ9@|4#+K0k}Z>-4&V z{{W5B{{U_1IsX7%;(|Zx;@|%Oh@rU)*rU^%QM7>KNHti!t}kt%{%M-a)qF+d<74 zy0Eox^`Bz6GfcxHKfD z^y0hk3t3rQcqm-z7qeRkj@XtMk(y^c7blPhUOoM)^hOhH^8Cm|OAL}kvaEp|fZ&WD zda(eBWQo~>kl#FhqwDWXyN(5h7qxjAU+)PqxG)1fJ7k<1i8vtDZj5$j#3}Y`iI4&p z1zRJhKjTHv-4Z(d+j!x7RJPflQnxI}7f}euyJKiS_t32SnI;l4O(U@f1Tplfj-|I^ z-OOf7dv^svWWn6wdSrL{R(r-2$g)VIV95#zJ-(mLt;dP9rcsU2NuHgUe~7AcDSMl9 z+Tt%L{{XxOKcK2Y-I*HI^iaVb(Yb&hd~uLZZ^EdL7pxAZ%*$%T{`NcxAbzbtD*q>!Pd|!d=R$mcPOU1Xf6G3jkhLM;VWItSC ze*kgsSh|$vYlmz&B=fxF=iJv5WXzWcK>4yOw73Mt8?s=sjFaA-I+u~1N_?wBI)9!l z=kA)qzG2crP>zImu7#8pU^)@Vu6tF=DIII67_H1n9}IcVy&}uaMIpmeH)6c%it;^b zdPc^brlmcc!b?fyhC%#D#ZT#5k@%JyqzkH9Cz<~MKUSbeC*r?#zaUSoasld8o7C1; zXsIPF$svh_yJyKc91Y(vsbW)Y2bfR>akK>><2`Eew2O%6vOvS{d4D1OYA4h4Ci^K) z_30Z75$#six4BA9N0!1{P5y^)RAF{)1M&X=>sLP^4JqJtu7unuv};Rk}hE&$~Y=Gb%4nDOkR}(;YM#LXlfUskRA45*r zj*<_hXr$Jt<&-tEBx#GBkUuI#emE4(t)5+JNBwqR&WOn^gyP=h@siEazbe+Yy<=)E zF4d*CA8?VDi#>Mdw>*xA*z;J6v^V6u`-Xg@MVO>1z0OfwwB$Ek6UeL$v9Ztyr642P9lcg=8oTx`K5kWzvfG4 zaV@j?(W;LxlA*E+nwIbjX`|Hi_98ZHl>?P;O02Ia`FnQ298`99Ce~+y;1dL#I6b`$ zQj}$6DMC^Ai_=@fG&`d!^KLj!}F(n4%5n%d*{w$fv# zr(!+nbsI9JS(q?o1tbHB7uV`L)>4a%jcLAU*rRUj8-Y^XI|;^lAaUQCuB`DtToK3C zr69YV-zjVkoYl+Tl&~D!N7x1*&aFO(nJ(t6*HCKiVxJJ)q1zS1umRL`M?Eq+2kp@OxGRx!43N(8 z%8ZgJA(e&*Bd9*~faS?2jP|SPGT%a$*KDaWF4!&@;1Ssz`gRO@bU!XO)L%CHiTk>A zEckLu!qTOSN*>&Or))59CvQ1p*+;%bZAS#rk!@;jvl2}#W++u$gBT={oQ21Jc&=91 zoj+4XxP+f7Oi2q8nZdzRjzx8F*~;*3RYsgg<^+b_iBZY<*|CM#NzeN^_pd^Oyli=R znA%qa7Oip}q)GM`*e_C`<^AnZ89 zcl-z>NQYW;Qa3EJNBr+8R**;?`gPag}ILCVQzX*ItlS$Fi z7qXJ#2;(doCEv7pn-qMXDEe2NOLY#Pr%eo(4>5?!(b>g$BBBB1s_qe~QgTV_kJ6I% zRgx(@^KSlMnxRo5bx^S@U~(Jp9QUUs87oN{*`}Obk3sRbjnh>(FJ%nY^Tz3I8B5iL zgoOV95}tVV?OZ(K;@bM*<2u|B#dhILG9}Dwa|&1)-T4aAY#1eP_nT*0YEO0=tn`8u#9^dEOPHI<_kXpK2b zS#|!rf>Ec2A87jux(hcVSSF0Na#?;}Imj${&jeLQPcvrtY%ZT;3`eMyF7igq8G{~5 zWDYpU$2qH}3)hYXM%D~w=F?*muA#8Ym0aNRxL(*8HHWEOmeS*qrM4|_#%#wFA=X85 zHsL;HQaSYYJ!>gS$mxRSc2D~G6}1((xr!1>$!`t1m}8Pe42VfM1Y>p%duKV~smvYi z)*swDpd_In5Pgr(RY+95$n$x@$k?bMq?B(1b^GN6%?GD~;IKDDBcGS{J$>L@+8JcZq+`4vQQFCkZ^ zf2sGV9kcSU-4)vWLGVS#gZ1e&xTBVJQIP8y{{VT_f0?HZGJoTe*+1U7 zPw}RN$8v|nW|$MjrRsQf(A zd=&}OlX+imx|J?oF}0;sSjwqxyK?7`ddJkE(SF-3T6nsQD-bu3@PMFXiNv&CKTuY>#wnxfrkw?k37wjxDuwC@T*;QY89 zaa>JBJMMP5N?L4u<9iE1GoydYHrg1B^zF@F@U5u18q^ZD5?wNO!yW@B9b53*`qKEK zCX2;7t-NDtOUXbP>4qYp(jDzB^t$pbeb|-*eri*W{BQG&a>N0S-iiQOl+WmhALMF1ZO1UgK$@7RUa%( zJ6l_1g(AGblPdceRy8clpaQ_|2ooPKP)2)`SB2ox?V*wx@f2vnFz_HQ8#{;`kWMj@ zPX{9viZwO1junJ1_)8Rx2kztCK4FuR+;Pc2l~s8qpEJ(InU2)<{{Zz@Pcm8@9F9)x zv3YwX+?$Wci^9eS08}Z@Sr}A%V^fcx^AzPVgUZN%;6PN~?A1jjWLZ@le));34oq?Z z*sAr(>^<>SQKH+Gh?DESu1eUK1p^(cjtT53k<`0kaTILENWqO%mQ+9(@yE=4>eP~p z#WO(Ip8=&+81pg7U_D20J;>uY&k}idWkgH8*bL(s0-tRj*-}}g@_if=kVhblbJx8W zC6=UjJI=T#M*7;d?QeGpM%e8}6#3lj({!+*^{{Nby_EdKd;pXrn;9 zJ35shdj%wu*zNVD(`QArDR{{u5BX&(mfjDhIbWzC^y@@c@?FT5c5uxqpeq=TJt z$nBhG*+3lPwXJ+F3=8&+Lgl=aQN3+|kmu09+#LE7QKwSu#kbJLwPuf_Fh*7|tuFYWl4M?tfSmXN8q|scdoQrs4#BlAC`y%DnAdD|s&%L#{ zvB1*j`05+u2>-*<3^$Fe7h5 zKk+N<8h67FS!Ar0NSX91ca@<_b$SChtARdaVLJg;NU zuE*XU=CGz1{5Y=5RnmNwdy84t?$;v|#t6YVz-2t4;P86j<0G2ouAnOHFh@NItmUQk0KJ;@iW42m6?*?l~PVW)vr!~4ZI!(Gut<5QdEvX!Q-V@wbUlJi5VMob0{wx z9EKx;KhI3ov^UYuFZx<6(Y6Zt#}W?M^8n8u?PPmZ^H}EPCbuR{B(mg5Yanoc**N}3 z`O~%42T=P$E)wAuRRgw1{{XJ8SVv};5~rCY1(lKExnGz5{x4s}K&drNHPgdZa<`Vo z2Y7eKBFL^j^5K+_Ps~O$+LUE!6yn;@$FctaStLL6)*=Vge8c%vSE^RbI(4_1WgkZW z0PBi;JELgi1Llo`3cYh|qG@0pq_Ca7>-T~D4O7vH9A2wxB)PeSW4ounf8|`Jt)`f5 z;<|_v9GM4?y2GgcNAa%O_bVl&H{$P29K4ETi;v?#PCpggKDeo< zu93AV@}_yl4M`CBg=)>DIAQ?keJfwXUKVXm^K6R66O)6C*A-c+QF50<(5AhsN+&lI zFDq?DW@2~%X1Yrc25Yy$8XZ1RxF0m1ApL9A^gj=2dQ{NM4Z1SC4Ee`9hiumMtS_is z#*#2;WE++RI7rvkyOZrvA8AovNUyw8oFSss{l}DAcn4IsnONQCzO+nN7+{ z%^pv4;d$2I&z9bJ*RP!ld9nT_NdmZCYRW{B!Z45n)qkk%Uh8SHyqF(-oyPfQBI@?27^(Mh7SG ztZ8ooTo1ANszJ)JY%_ixht`y5%&iB_=)X6ymn5OSpF5RfR=$($w<{kv6Ej!n=&f+)ED5l*Gkjk=1H=5kMDU~P~4%0B}@z&uI^;Yb{RcRz&# z?{xht?2PWm=N%>=g3^~b{{Ub58Y2X2I^{pXv?u_P>R%Z2%i#`41G1<=#modUgB!Nd> zhdmFqXls5AxU_a@uIyA6!pJ4UH?9e+?+fbp8aAr85VU4IktuGCdO3&1N(Tfb1j_*9Wa`=~nW1h)R{XqS`! z00JlJPBfz8P z)77{X(#wuA-nba2K694kz3L&HAFVyWac#@bK7do;&f;;5DLp-^Cf&F1dUvVqK=S4% zvU>Z9q(duU?jKrq?oN7pQ_2NR!Ph&njMEiKUfLuK<`(@+iOWPa%V>0A&3bjaZ6liS|8`QjqtL{#1BgR*gxh0*# zt*Tlj)HdSg6}M6)?UPh{|;+@#S*ENo*pdsV!+x1Jm38Qwu1jzvOaR2GRUI;2s=-ej&B4-WlG@xcSH z6*gsE5MLNtJI%{%T!$$&5vVFpG;r1l$O#|+}rF6IZWQ`)dyP7yIw<-@ZJBYV#^FP*Z8!%kwjEs8!ks*%Mt%O4^+ zC2aJKN&Z#Vc)Ay`)-Ivbt*z1+q#=T$%2el`c?0mSQhV#k#wNLy$7S4bKMLlYd0m<4 z)0Z@Pmr`%+yQTjCUL&^{{{WWqs-Md_{VBg^w2%F5`nmMCg`f3pwMvv8YBVm-xpU64 zU%3SGpIIde58wuWS_Yz1n|oVmzwMs?0R0a&PwurT^&RQmiX_X?OEmN}%D9x#xpf1N~V40D6#~57wVE7bIg<0SPpPD$fs2o7p4QH5xqP@2dt)cC zQR+`XJz~sdcCqC@mN&3|i-(khA zH1C$*ZRN=8D=uSC_f-Ak?s)d^Rm{&Mk0j30O8r=znsBv9zvJTKD9`sn-rtb^w4zHm z@wOSJZ&?^1{Dx2KOq~a_GRBgj#zxY8#~;fdoo#vY&EZ=xjDK@^KbXUy-jQt?_ww71 zYRXMQMYC@qQrw3de6{3y@^XJ#+0lGQ6|J4lo2uER-G%1UDF+g(u6D?*bMp<&^$bDl zTBoUulWSs>Ya>W9!200VzI-~KOD}=C#nrnNnkKn*W#EPpt_^sN&Y1T0{{U{WlwMeq zk2*1f89f$7>WB0_iLYh&J*(*Vo(9#l9cJk*5=kA4OEO2fBtWNWC#WOv$gO4YocH&O z{sr+Z=CR-(6j^Gw4RIx{t4NYa*mMh`?C7xDYyP$x^TfPf2=M_?t<&unErZ9_g zgr_C$_Pacf#1Dqhp zT=gd{PExci&7!X@4s@vvw`s`Z0+Fzv>j)?NqxzcCYw2zYZAKrpGj))ivVC`sr|X)i z_7J<~)^bn$^P336@%yLnAp2J{OzKiyN{r>Y{sYp5jGjoVFK;EQVrZH+JvXTt{OW6K zSuPoZ4a^XbJ#k4U*f}JwS4i?bs?17pwKxZ<9qVRK3|?DI(A>#o6w#8c8(k(r9lo^B z?3u)Y7Hd{H#?_n?pH3@Oj3qS6kwRUTIPsI-reYp!N6JJnzW()c?@PVX7(sb%(90}n z;O8S_C}Gdgiee(#AO&~I+<-b;l^rYfph!A+wQGN3{pr)ja#3* zamoB@+Q4laXi)b#z){M_$6s?)mCWTNh6ollUompWm55}JJzM-L+pBISRuQWn6-Ug} zLuly5h8|PPyPPV0O(+pXmsuV{crq&WK8v5{RA;#Kj1?%62^mgy?)g*Pnvf(WFw#h0 zX9Pf)2kOS7Cp%-ASod34`jhI$q*Rm5R1qW3*$V=Z{sa2ZBS%SC-%+%Rv<*8zYi4WIX=a1>Qb8G5op}Yiu*t|gU=K`J7c5rk^Ij8oCV2{W zp!U!BRK)0q!$NCwqfx)@X+Q!b=+qSS0<+<>hp6cP2<& zAABG<{{U*Jpp39om&{}QF~ionj}Ge^WbxT)x4LXiZ^{&4o@+>3?*?VUfC(o&aC6qP zHHp^J`e-z}mNsg|V`w5&GAQFDj!u1gdkVGK&Pc2FIU)hBo;xJ}0H&BSAEFL_I_Y$; z9YNv!CG2eAHxbC&DiAy+o;F+$;SYoUJXO1k$YIkoEmd~h+DUA2 z6LPCEaKMjIkZX*wxzwyK&CY>v`k(Ha3yn7MN1NrJy%|t{c)#s0JYeq;1d;KQYUxjvDA@Z(Daq|w1{(VI_{6?{5Rg+0uhfoKY(u|+{cBx%>;yJY2 z>sQlIMRE?&zliu4Jv$MaJ@(V({D&W98?I~L$?wp>xhzG)rv~ON&Idxk=RaDN?A9%F z!uQtBRJvR5E5{7`9Dbx#Nv~{Sw~pfeZ?ap3z}#}v9CNgOeDjRdEfxN=HLdiFz8ui4 zdS2cgjLJCy6!kuoO{KU&H!TYOH_>ey#2Sv91-i#2?wQJnIOI#ky%gImEx<{g#t0X?0@x^S*3gelhQ|Lh-(yK`hV<#Y$IR$&? z{{Yoc<8ye-$T6QvW(uA-tr=j)%ou#bH4DTCyLSLqPBu1DD((&U#t81e;YqhYbcXsG zdq#aJo>OOfm9 zs`zf>Mj(eMObLdgW{)TG<5@#FR^EFKq7j!cX{nPnV!nx>;`B8~K zyxjZKmNsGcoAabQiXo8lP^k58Do_<8WkbD~V7EX%l`K|?Dh;O+cUL$K^gX_Z^X*c- zoRKhDK}SEr>Oubi>(f^u$rWRZ<{9r(#~1GN{nqBLEu@8b12|GXVc1p3^!7ey-2vC- z%dcGIcdJn{l|)y!g!yEF%uXBdJb#6B7O@ER7>VLJlmdML;-j|H9b`PmvPep|?^lKc zBfqE>eJtmDI9Nu-<2yiJc*#AhRb+EYN!-0{88#9jA2bh>>^dLU6>c-W*nfF?`-*kF zyzPBB@+H|kZIiI|#_#3@5!+lKIk%K{{{U8Mv@+DXw=+fdTgYT(c=H<&amF$K0HBFc%u@P?i zk!t#Fv|nkqyAoTFeCVSnRg|0pFnfQ1tkjVp)g!yIW;cje3{fqA3lnDA8~_GD(SNl&kXhqrGv9L+w*S#f@Zh*h`~E@ zzj$$uqPX2wC$_Yg&PGB82**S3=~F6ktqw^WLhzDk^t~Tmu)bM~Tttzig@oJ`!Qk}i z&m?=2F^n4)*Hh_N5Zx~;jfprr;*ff799Mu_>UVl==bvM9Exe=n;&P|-u9nB*4a`|< zn^b98cD(j-8)+OK%m66gis#<2jp;61T^&^+?LFn8^c|h*$sDuUtXC6E2JO;)-)v%; z1L{%YCyp;K%@P~|UK?rSHJ?yPP3yvUgSFvQ{5J;!@xZR=c%xBk80ELpFRze{dDic5 zB9rw^^gdf}_nBI0>-n8r&^#*QcZ37qetYM)(xZ%-?qrfBaRxADDULn8>yDeonyd@D zZ6_r`zU)aO@v7yETEDyc{Y7Vc z+xq^zk43hH3AtcX2ETvJ(tu zMck-PK-;*GIrTqU>@K`jeQBZj*0&a39@6Fn#~d@6uC9-8SOUepa#cogn)5mDP05!l zm1iuj>Nk;|dwi)OH&dD2cnCg7(0XV5D<<<-F2PUE0Lr4K5K`uh zJ=k>ZuyQ$j&3xImw-Ca)gtV&14JI+a*dO5^PHP_YHN<7nasl~@j4F@wide3e=EY=_ z&YEOweCz@s_8m=U-pM41mu@=@U{qCti(JoBdJ>eP)E1i>T6VQHy|~k0R9NSfG>?y% z_Sz3&Q+Uoet@R6dAOIAIZzIC#E#DL&N!^yVBEp>fBUaYGJ!}S z3aZC*6YuFtqP)@ydQ)$?B7C53$?p;CoXozVO^K$L@pM-lo6((<~!` z2hD@tu5HFTZpeaDmb)Ue%0>$OxaombY;N`|MK)O+Mqqe68zUFN3e|G{(5Pgsg(yJlLX_=8Q^s547+|>-bkSYjGcz zVYv*^XOagg{?PuTxcms|d^s78(@;{|*UAj6y$IYt#w<7Thm2Ra%i*sMYC1@U+eovD-M-*kRSJ0P zfInWf`K6-LL!(#xi+L#y`j&@q!=^EnZp4g^MRLPCbCNl( z-^7H;@b5!E^Uhn8$NULT^{ylX#$%$~vXpxp{8ZyE{{ZGSfFN_<(yUvJwv!)HqM%@S z%Hx`}gKnBnP157^t6vK;YVt^fcO)?-LCF=r;lBy$n&6t|!p>>d+-_oscA@RK^dJ3d z!ip!FRakIXoc%q|u&-M1H^vKJ1Za0UYucFrjm#z`A3THr8NXW6oK>#Ot4>jAMRog+ zJ=C=I@U78;{0X%|$+sgb0sjDR)xEk3$dYO1xM}BhX&PQYtb<7;vOH&IOLWM`P&WX3?d0Pg z;=n-j6q8U3xDb5XfbWR|hCW#!0z!aB?2nhdb5>fMHaCjw@RdI+foX%Z^}sx4yRWlB zG#+-BsTqdC?d916i#~yZ!U>)6Y)~`07cq8&+k%UVu zaEFT8J;ha7vPzupJBc~%&TBI53`Wu|QDmMdxguDKi7}RBEKf{;2cW7=KE@qK${sjL zx`0nDoS;Vy_2WtAC}zqkC>?R0xZwW)dbw}y`K{x4HnYt(VIj@&gE%KX}v z^(%=aky2@)k)u$%;2bamgTNg6=bu_zZ9(m1w^em_{cAUI=z3DjbF&_R zS1&ug&qI+VV@PU#V~BMhG0J?YGBzY%$G_0jbJ^}gERM}PAb zS)|7(_*sA8E(gu@BR^V*h@~4=Liqj()b3xzQswL% mHhaE>s0r_g#vHm1tzUSs2 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 dt&$$MvVMDu$A}%>>>>>>> EXECUTING FUNCTION get_weather... user_proxy (to chatbot): ***** Response from calling function "get_weather" ***** 112.22727272727272 EUR **************************************************************** + +``` + +Once the initial chat is processed, the Agent will send the suggest tool to call. In this case, it is a function called `get_weather`. Depending on your configuration, this function can be automatically executed and read by the Agent or can be executed based on user input. + +We will look at this process in more detail with the Python example included in this lesson. + +## Taskweaver + +The next agent framework we will explore is Taskweaver. It is known as a "code-first" agent because instead of working strictly with `strings` , it can work with dataframes in Python. This becomes extremely useful for data analysis and generation tasks. This can be things like creating graphs and charts or generating random numbers. + +### State and Tools + +To manage the state of the conversation, TaskWeaver uses the concept of a `Planner`. The `Planner` is a LLM that takes the request from the users and maps out the tasks that need to be completed to fulfill this request. + +To complete the tasks the `Planner` is exposed to the collection of tools called `Plugins`. This can be Python classes or a general code interpreter. This plugins are stored as embeddings so that the LLM can better search for the correct plugin. + +Here is an example of a plugin to handle anomaly detection: + +```python +class AnomalyDetectionPlugin(Plugin): def __call__(self, df: pd.DataFrame, time_col_name: str, value_col_name: str): +``` + +The code is verified before executing. Another feature to manage context in Taskweaver is `experience`. Experience allows for the context of a conversation to be stored over to the long term in a YAML file. This can be configured so that the LLM improves over time on certain tasks given that it is exposed to prior conversations. + +## JARVIS + +The last agent framework we will explore is JARVIS. What makes JARVIS unique is that it uses an LLM to manage the `state` of the conversation and the `tools`are other AI models. Each of the AI models are specialized models that perform certain tasks such as object detection, transcription or image captioning. + +The LLM, being a general purpose model, receives the request from the user and identifies the the specific task and any arguments/data that is needed to complete the task. + +```python +[{"task": "object-detection", "id": 0, "dep": [-1], "args": {"image": "e1.jpg" }}] +``` + +The LLM then formats the request in a manner that the specialized AI model can interpret, such as JSON. Once the AI model has returned its prediction based on the task, the LLM receives the response. + +If multiple models are required to complete the task, it will also interpret the response from those models before bringing them together to generate to the response to the user. + +The example below shows how this would work when a user is requesting a description and count of the objects in a picture: + +## Assignment + +To continue your learning of AI Agents you can build with AutoGen: + +- An application that simulates a business meeting with different departments of an education startup. +- Create system messages that guide LLMs in understanding different personas and priorities, and enable the user to pitch a new product idea. +- The LLM should then generate follow-up questions from each department to refine and improve the pitch and the product idea diff --git a/README.md b/README.md index be477855e..4174d8cf3 100644 --- a/README.md +++ b/README.md @@ -1,6 +1,6 @@ ![Generative AI For Beginners](./images/repo-thubmnail2.png?WT.mc_id=academic-105485-koreyst) -### 18 Lessons teaching everything you need to know to start building Generative AI applications +### 18 Lessons teaching everything you need to know to start building Generative AI applications [![GitHub license](https://img.shields.io/github/license/microsoft/Generative-AI-For-Beginners.svg)](https://github.com/microsoft/Generative-AI-For-Beginners/blob/master/LICENSE?WT.mc_id=academic-105485-koreyst) [![GitHub contributors](https://img.shields.io/github/contributors/microsoft/Generative-AI-For-Beginners.svg)](https://GitHub.com/microsoft/Generative-AI-For-Beginners/graphs/contributors/?WT.mc_id=academic-105485-koreyst) @@ -15,74 +15,81 @@ [![](https://dcbadge.vercel.app/api/server/ByRwuEEgH4)](https://aka.ms/genai-discord?WT.mc_id=academic-105485-koreyst) **Version 2 of this course is currently being deployed. Official Release February 23rd. Pardon the mess meanwhile** -# Generative AI for Beginners - A Course -Learn the fundamentals of building Generative AI applications with our 18-lesson comprehensive course by Microsoft Cloud Advocates. +# Generative AI for Beginners - A Course + +Learn the fundamentals of building Generative AI applications with our 18-lesson comprehensive course by Microsoft Cloud Advocates. + ## 🌱 Getting Started -This course is 18 lessons. Each lesson covers its own topic so start wherever you would like! +This course is 18 lessons. Each lesson covers its own topic so start wherever you would like! + +Lessons are labeled either "Learn" lessons explaining a Generative AI concept or "Build" lessons that explain a concept and code examples in both **Python** and **TypeScript** when possible. -Lessons are labeled either "Learn" lessons explaining a Generative AI concept or "Build" lessons that explain a concept and code examples in both **Python** and **TypeScript** when possible. +Each lesson also includes a "Keep Learning" section with additional learning tools. -Each lesson also includes a "Keep Learning" section with additional learning tools. +**What You Need** -**What You Need** -- Access to the [Azure OpenAI Service](https://azure.microsoft.com/en-us/products/ai-services/openai-service?WT.mc_id=academic-105485-koreyst) **OR** [OpenAI API](https://platform.openai.com/docs/quickstart?context=python?WT.mc_id=academic-105485-koreyst) - *Only required to complete coding lessons* -- Basic knowledge of Python or Typescript is helpful - *For absolute beginners check out these [Python](https://learn.microsoft.com/en-us/training/paths/python-language/?WT.mc_id=academic-105485-koreyst) and [TypeScript](https://learn.microsoft.com/en-us/training/paths/build-javascript-applications-typescript/?WT.mc_id=academic-105485-koreyst) courses. +- Access to the [Azure OpenAI Service](https://azure.microsoft.com/en-us/products/ai-services/openai-service?WT.mc_id=academic-105485-koreyst) **OR** [OpenAI API](https://platform.openai.com/docs/quickstart?context=python?WT.mc_id=academic-105485-koreyst) - _Only required to complete coding lessons_ +- Basic knowledge of Python or Typescript is helpful - \*For absolute beginners check out these [Python](https://learn.microsoft.com/en-us/training/paths/python-language/?WT.mc_id=academic-105485-koreyst) and [TypeScript](https://learn.microsoft.com/en-us/training/paths/build-javascript-applications-typescript/?WT.mc_id=academic-105485-koreyst) courses. - A Github Account to [fork this entire repo](https://github.com/microsoft/generative-ai-for-beginners/fork?WT.mc_id=academic-105485-koreyst) to your own GitHub account -We have created a **[Course Setup](./00-course-setup/README.md?WT.mc_id=academic-105485-koreyst)** lesson to help you with setting up your developement environment. +We have created a **[Course Setup](./00-course-setup/README.md?WT.mc_id=academic-105485-koreyst)** lesson to help you with setting up your developement environment. -Don't forget to star [star (🌟) this repo](https://docs.github.com/en/get-started/exploring-projects-on-github/saving-repositories-with-stars?WT.mc_id=academic-105485-koreyst) to find it easier later. +Don't forget to star [star (🌟) this repo](https://docs.github.com/en/get-started/exploring-projects-on-github/saving-repositories-with-stars?WT.mc_id=academic-105485-koreyst) to find it easier later. -## 🧠 Ready to Deploy? -If you are looking for more advanced code samples, check out our [collection of Generative AI Code Samples](https://aka.ms/genai-beg-code) in both **Python** and **TypeScript**. +## 🧠 Ready to Deploy? -## 🗣️ Meet Other Learners, Get Support +If you are looking for more advanced code samples, check out our [collection of Generative AI Code Samples](https://aka.ms/genai-beg-code) in both **Python** and **TypeScript**. - Join our [official AI Discord server](https://aka.ms/genai-discord?WT.mc_id=academic-105485-koreyst) to meet and network with other learners taking this course and get support. +## 🗣️ Meet Other Learners, Get Support -## 🚀 Building a Startup? +Join our [official AI Discord server](https://aka.ms/genai-discord?WT.mc_id=academic-105485-koreyst) to meet and network with other learners taking this course and get support. -Sign up for [Microsoft for Startups Founders Hub](https://aka.ms/genai-foundershub?WT.mc_id=academic-105485-koreyst) to receive **free OpenAI credits** and up to **$150k towards Azure credits to access OpenAI models through Azure OpenAI Services**. +## 🚀 Building a Startup? -## 🙏 Want to help? +Sign up for [Microsoft for Startups Founders Hub](https://aka.ms/genai-foundershub?WT.mc_id=academic-105485-koreyst) to receive **free OpenAI credits** and up to **$150k towards Azure credits to access OpenAI models through Azure OpenAI Services**. + +## 🙏 Want to help? + +Find spelling errors, code errors or have a suggestion? [Raise an issue](https://github.com/microsoft/generative-ai-for-beginners/issues?WT.mc_id=academic-105485-koreyst) or [Create a pull request](https://github.com/microsoft/generative-ai-for-beginners/pulls?WT.mc_id=academic-105485-koreyst) -Find spelling errors, code errors or have a suggestion? [Raise an issue](https://github.com/microsoft/generative-ai-for-beginners/issues?WT.mc_id=academic-105485-koreyst) or [Create a pull request](https://github.com/microsoft/generative-ai-for-beginners/pulls?WT.mc_id=academic-105485-koreyst) ## 📂 Each lesson includes: - A short video introduction to the topic -- A written lesson located in the README -- Python and TypeScript Code Samples supporting Azure OpenAI and OpenAI API +- A written lesson located in the README +- Python and TypeScript Code Samples supporting Azure OpenAI and OpenAI API - Links to extra resources to continue your learning ## 🗃️ Lessons -| | Lesson Link | Description | Additional Learning | -| :--: | :--: | :--: | ---- | -| 00 | [Course Setup](./00-course-setup/README.md?WT.mc_id=academic-105485-koreyst) | **Learn:** How to Setup Your Development Environment | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 01 | [Introduction to Generative AI and LLMs](./01-introduction-to-genai/README.md?WT.mc_id=academic-105485-koreyst) | **Learn:** Understanding what Generative AI is and how Large Language Models (LLMs) work. | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 02 | [Exploring and comparing different LLMs](./02-exploring-and-comparing-different-llms/README.md?WT.mc_id=academic-105485-koreyst) | **Learn:** How to select the right model for your use case | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 03 | [Using Generative AI Responsibly](./03-using-generative-ai-responsibly/README.md?WT.mc_id=academic-105485-koreyst) | **Learn:** How to build Generative AI Applications responsibly | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 04 | [Understanding Prompt Engineering Fundamentals](./04-prompt-engineering-fundamentals/README.md?WT.mc_id=academic-105485-koreyst) | **Learn:** Hands-on Prompt Engineering Best Practices | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 05 | [Creating Advanced Prompts](./05-advanced-prompts/README.md?WT.mc_id=academic-105485-koreyst) | **Learn:** How to apply prompt engineering techniques that improve the outcome of your prompts. | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 06 | [Building Text Generation Applications](./06-text-generation-apps/README.md?WT.mc_id=academic-105485-koreyst) | **Build:** A text generation app using Azure OpenAI | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 07 | [Building Chat Applications](./07-building-chat-applications/README.md?WT.mc_id=academic-105485-koreyst) | **Build:** Techniques for efficiently building and integrating chat applications. | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 08 | [Building Search Apps Vector Databases](./08-building-search-applications/README.md?WT.mc_id=academic-105485-koreyst) | **Build:** A search application that uses Embeddings to search for data. | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 09 | [Building Image Generation Applications](./09-building-image-applications/README.md?WT.mc_id=academic-105485-koreyst) | **Build:** A image generation application | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 10 | [Building Low Code AI Applications](./10-building-low-code-ai-applications/README.md?WT.mc_id=academic-105485-koreyst) | **Build:** A Generative AI application using Low Code tools | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 11 | [Integrating External Applications with Function Calling](./11-integrating-with-function-calling/README.md?WT.mc_id=academic-105485-koreyst) | **Build:** What is function calling and its use cases for applications | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 12 | [Designing UX for AI Applications](./12-designing-ux-for-ai-applications/README.md?WT.mc_id=academic-105485-koreyst) | **Learn:** How to apply UX design principles when developing Generative AI Applications | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 13 | Securing Your Generative AI Applications | **Learn:** The threats and risks to AI systems and methods to secure these systems. | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 14 | The Generative AI Application Lifecycle | **Learn:** The tools and metrics to manage the LLM Lifecycle and LLMOps | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 15 | Retrieval Augmented Generation (RAG) and Vector Databases | **Build:** An application using a RAG Framework to retrieve embeddings from a Vector Databases | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 16 | Open Source Models and Hugging Face | **Build:** An application using open source models available on Hugging Face | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 17 | AI Agents and AutoGen | **Build:** An application using an AI Agent Framework | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | -| 18 | Fine-Tuning LLMs | **Learn:** The what, why and how of fine-tuning LLMs | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | - -### 🌟 Special thanks -Special thanks to [**John Aziz**](https://www.linkedin.com/in/john0isaac/) for creating all of the GitHub Actions and workflows - -## 🎒 Other Courses + +| | Lesson Link | Description | Additional Learning | +| :-: | :------------------------------------------------------------------------------------------------------------------------------------------: | :---------------------------------------------------------------------------------------------: | ------------------------------------------------------------------------------ | +| 00 | [Course Setup](./00-course-setup/README.md?WT.mc_id=academic-105485-koreyst) | **Learn:** How to Setup Your Development Environment | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 01 | [Introduction to Generative AI and LLMs](./01-introduction-to-genai/README.md?WT.mc_id=academic-105485-koreyst) | **Learn:** Understanding what Generative AI is and how Large Language Models (LLMs) work. | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 02 | [Exploring and comparing different LLMs](./02-exploring-and-comparing-different-llms/README.md?WT.mc_id=academic-105485-koreyst) | **Learn:** How to select the right model for your use case | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 03 | [Using Generative AI Responsibly](./03-using-generative-ai-responsibly/README.md?WT.mc_id=academic-105485-koreyst) | **Learn:** How to build Generative AI Applications responsibly | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 04 | [Understanding Prompt Engineering Fundamentals](./04-prompt-engineering-fundamentals/README.md?WT.mc_id=academic-105485-koreyst) | **Learn:** Hands-on Prompt Engineering Best Practices | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 05 | [Creating Advanced Prompts](./05-advanced-prompts/README.md?WT.mc_id=academic-105485-koreyst) | **Learn:** How to apply prompt engineering techniques that improve the outcome of your prompts. | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 06 | [Building Text Generation Applications](./06-text-generation-apps/README.md?WT.mc_id=academic-105485-koreyst) | **Build:** A text generation app using Azure OpenAI | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 07 | [Building Chat Applications](./07-building-chat-applications/README.md?WT.mc_id=academic-105485-koreyst) | **Build:** Techniques for efficiently building and integrating chat applications. | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 08 | [Building Search Apps Vector Databases](./08-building-search-applications/README.md?WT.mc_id=academic-105485-koreyst) | **Build:** A search application that uses Embeddings to search for data. | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 09 | [Building Image Generation Applications](./09-building-image-applications/README.md?WT.mc_id=academic-105485-koreyst) | **Build:** A image generation application | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 10 | [Building Low Code AI Applications](./10-building-low-code-ai-applications/README.md?WT.mc_id=academic-105485-koreyst) | **Build:** A Generative AI application using Low Code tools | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 11 | [Integrating External Applications with Function Calling](./11-integrating-with-function-calling/README.md?WT.mc_id=academic-105485-koreyst) | **Build:** What is function calling and its use cases for applications | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 12 | [Designing UX for AI Applications](./12-designing-ux-for-ai-applications/README.md?WT.mc_id=academic-105485-koreyst) | **Learn:** How to apply UX design principles when developing Generative AI Applications | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 13 | [Securing Your Generative AI Applications](./13-securing-ai-applications/README.md?WT.mc_id=academic-105485-koreyst) | **Learn:** The threats and risks to AI systems and methods to secure these systems. | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 14 | [The Generative AI Application Lifecycle](./14-the-generative-ai-application-lifecycle/README.md?WT.mc_id=academic-105485-koreyst) | **Learn:** The tools and metrics to manage the LLM Lifecycle and LLMOps | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 15 | [Retrieval Augmented Generation (RAG) and Vector Databases](<(./15-rag-and-vector-databases/README.md?WT.mc_id=academic-105485-koreyst)>) | **Build:** An application using a RAG Framework to retrieve embeddings from a Vector Databases | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 16 | Open Source Models and Hugging Face | **Build:** An application using open source models available on Hugging Face | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 17 | [AI Agents](./17-ai-agents/README.md?WT.mc_id=academic-105485-koreyst) | **Build:** An application using an AI Agent Framework | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | +| 18 | Fine-Tuning LLMs | **Learn:** The what, why and how of fine-tuning LLMs | [Learn More](https://aka.ms/genai-collection?WT.mc_id=academic-105485-koreyst) | + +### 🌟 Special thanks + +Special thanks to [**John Aziz**](https://www.linkedin.com/in/john0isaac/) for creating all of the GitHub Actions and workflows + +## 🎒 Other Courses Our team produces other courses! Check out: