From 8e1a04414fd8761eacb83216ea798d6647d5b88d Mon Sep 17 00:00:00 2001 From: Marco Cetraro Date: Mon, 21 Oct 2024 21:36:46 +0000 Subject: [PATCH] update changes --- .../gcp_rag_structure_data_01-checkpoint.png | Bin 0 -> 16270 bytes ...CP_RAG_for_Structure_Data-checkpoint.ipynb | 598 +++++++++++++++ ...AG_for_Structure_Data_old-checkpoint.ipynb | 710 ++++++++++++++++++ .../GenAI/GCP_RAG_for_Structure_Data.ipynb | 160 +--- .../GCP_RAG_for_Structure_Data_old.ipynb | 710 ++++++++++++++++++ 5 files changed, 2042 insertions(+), 136 deletions(-) create mode 100644 images/.ipynb_checkpoints/gcp_rag_structure_data_01-checkpoint.png create mode 100644 notebooks/GenAI/.ipynb_checkpoints/GCP_RAG_for_Structure_Data-checkpoint.ipynb create mode 100644 notebooks/GenAI/.ipynb_checkpoints/GCP_RAG_for_Structure_Data_old-checkpoint.ipynb create mode 100644 notebooks/GenAI/GCP_RAG_for_Structure_Data_old.ipynb diff --git a/images/.ipynb_checkpoints/gcp_rag_structure_data_01-checkpoint.png b/images/.ipynb_checkpoints/gcp_rag_structure_data_01-checkpoint.png new file mode 100644 index 0000000000000000000000000000000000000000..a9cedd885c6c716900c5734a397876b5f30f8ff5 GIT binary patch literal 16270 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" + ] + }, + { + "cell_type": "markdown", + "id": "6c0533e3-4fbd-47b1-8458-fd193bcb263f", + "metadata": {}, + "source": [ + "# Creating a Chatbot for Structure Data using a RAG" + ] + }, + { + "cell_type": "markdown", + "id": "76cf4b02-6422-4a48-882c-c00315e34c8a", + "metadata": { + "tags": [] + }, + "source": [ + "## Overview" + ] + }, + { + "cell_type": "markdown", + "id": "7b7870dd-09e0-4f6a-91a9-7c30529af720", + "metadata": {}, + "source": [ + "Generative AI (GenAI) represents a transformative technology capable of producing human-like text, images, code, and various other types of content. While much of the focus has been on unstructured data—such as PDFs, text documents, image files, and websites—many GenAI implementations rely on a parameter called \"top K.\" This algorithm retrieves only the highest-scoring pieces of content relevant to a user's query, which can be limiting. Users seeking insights from structured data formats like CSV and JSON often require access to all relevant occurrences, rather than just a subset.\n", + "\n", + "In this tutorial, we use a RAG agent together with a VertexAI chat model gemini-1.5-pro to query the BigQuery table. A Retrieval Augmented Generation (RAG) agent is a key part of a RAG application that enhances the capabilities of the large language models (LLMs) by integrating external data retrieval. AI agents empower LLMs to interact with the world through actions and tools." + ] + }, + { + "cell_type": "markdown", + "id": "d513a9fe-1ac4-4610-b53c-6696e461f828", + "metadata": {}, + "source": [ + "## Prerequisites" + ] + }, + { + "cell_type": "markdown", + "id": "fbdb09aa-73d6-4cee-b3e7-5d6b243e3d58", + "metadata": {}, + "source": [ + "We assume you have access to Vertex AI and have enabled the necessary APIs.\n", + "\n", + "In this tutorial, we will be using Google Gemini Pro 1.5, which does not require deployment. However, if you prefer to use a different model, you can select one from the Model Garden via the console. This will allow you to add the model to your registry, create an endpoint (or utilize an existing one), and deploy the model—all in a single step. Here is a link for more information: [model deployment](https://cloud.google.com/vertex-ai/docs/general/deployment).\n", + "\n", + "Before we begin, you'll need to create a Vertex AI RAG Data Service Agent service account. To do this, go to the IAM section of the console. Ensure you check the box for \"Include Google-provided role grant.\" If the role is not listed, click \"Grant Access\" and add \"Vertex AI RAG Data Service Agent\" as a role.\n", + "\n", + "We are using a e2-standard-4 (Efficient Instance: 4 vCPUs, 16 GB RAM) for this tutorial. " + ] + }, + { + "cell_type": "markdown", + "id": "eeb24716-0db8-462f-8faa-0b8ad9d61feb", + "metadata": { + "tags": [] + }, + "source": [ + "## Learning objectives" + ] + }, + { + "cell_type": "markdown", + "id": "f5216a43-31ee-4c3e-821a-4b81f4be8e54", + "metadata": {}, + "source": [ + "In this tutorial you will learn about:\n", + "- How to set up a BigQuery dataset and a table.\n", + "- How to load data to a BigQuery table.\n", + "- How to use the langchain ChatVertexAI agent to extract information from the table. \n" + ] + }, + { + "cell_type": "markdown", + "id": "348910aa-9802-4318-8fcb-86603158923b", + "metadata": { + "tags": [] + }, + "source": [ + "## Pricing" + ] + }, + { + "cell_type": "markdown", + "id": "0dca31b2-880d-4cbc-bf76-ede96998e371", + "metadata": {}, + "source": [ + "If you are following this tutorial in one sitting it will cost $23.89 per month. Completing the process in multiple sessions or using a method different from the tutorial may result in increased costs." + ] + }, + { + "cell_type": "markdown", + "id": "c7d9e3ee-1c7e-4baa-a818-c13330c665dd", + "metadata": {}, + "source": [ + "## Get Start" + ] + }, + { + "cell_type": "markdown", + "id": "137defcd-fd2b-421d-9b2e-fe2b878fb325", + "metadata": {}, + "source": [ + "### Install Packages" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "55c14a38-ba51-466d-90b5-1512f5a7fae0", + "metadata": { + "tags": [] + }, + "outputs": [], + "source": [ + "!pip install google-cloud-bigquery\n", + "!pip install gcloud\n", + "!pip install langchain\n", + "!pip install -U langchain-community\n", + "!pip install -qU langchain_google_vertexai\n", + "!pip install pandas-gbq" + ] + }, + { + "cell_type": "markdown", + "id": "64e22773-9a03-47de-9b6d-7946abcba722", + "metadata": {}, + "source": [ + "Set your project id, location, and bucket variables." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "7193a9ac-3169-45c1-9c5d-ba5e2f31458f", + "metadata": {}, + "outputs": [], + "source": [ + "project_id=''\n", + "location=' (e.g.us-east4)'\n", + "bucket = ''" + ] + }, + { + "cell_type": "markdown", + "id": "df0d9eda-ee54-4c4c-86e6-9166b2b82e2d", + "metadata": {}, + "source": [ + "Create a bucket where the data used for tutorial is going to be placed." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "7b583234-2d92-4e5b-841d-c7d79dd87d45", + "metadata": {}, + "outputs": [], + "source": [ + "# make bucket\n", + "!gsutil mb -l {location} gs://{bucket}" + ] + }, + { + "cell_type": "markdown", + "id": "5781a07c-84ed-416c-a782-b1dc881e5c18", + "metadata": {}, + "source": [ + "Once the bucket is created, we need to access the CSV source file. In this tutorial, I transferred the data file to our Jupyter notebook by simply dragging and dropping it from my local folder. Next, we need to specify the bucket name and the path of the data source in order to upload the CSV file to the bucket. It is important to keep in mind that the name of the bucket has to be unique." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "4829094e-61a6-4dd7-854f-6b21dc19b1b7", + "metadata": {}, + "outputs": [], + "source": [ + "!gsutil cp '' gs://{bucket}" + ] + }, + { + "cell_type": "markdown", + "id": "7a00dc83-8713-4ea5-9345-08a9f3d785d0", + "metadata": {}, + "source": [ + "The next step is to create a database in BigQuery. To accomplish this, we need to define a dataset_id and construct a dataset object that will be sent to the API for creation. Note that in BigQuery, a dataset is analogous to a database." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "1750e13d-ec05-4335-82a2-ffe54c5b5f32", + "metadata": {}, + "outputs": [], + "source": [ + "from google.cloud import bigquery\n", + "\n", + "client = bigquery.Client()\n", + "\n", + "dataset_name = ''\n", + "# Set dataset_id to the ID of the dataset to create.\n", + "dataset_id = f'{project_id}.{dataset_name}'\n", + "\n", + "# Construct a full Dataset object to send to the API.\n", + "dataset = bigquery.Dataset(dataset_id)\n", + "\n", + "# Specify the geographic location where the dataset should reside.\n", + "dataset.location = location\n", + "\n", + "# Send the dataset to the API for creation, with an explicit timeout.\n", + "# Raises google.api_core.exceptions.Conflict if the Dataset already\n", + "# exists within the project.\n", + "dataset = client.create_dataset(dataset, timeout=30) # Make an API request.\n", + "print(\"Created dataset {}.{}\".format(client.project, dataset.dataset_id))" + ] + }, + { + "cell_type": "markdown", + "id": "544c191c-2bc0-4a47-a0a7-cea81035597a", + "metadata": {}, + "source": [ + "Once the dataset is created, the next step is to create a table. First, we define a table ID, followed by outlining the table's schema. Finally, we send an API request to create the table within the dataset established in the previous step." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "68206b14-1b5e-49c3-9f07-52ba6c70a93e", + "metadata": {}, + "outputs": [], + "source": [ + "# Set table_id to the ID of the table to create.\n", + "\n", + "client = bigquery.Client()\n", + "\n", + "table_name = \"\"\n", + "table_id = f'{project_id}.{dataset_id}.{table_name}'\n", + "\n", + "schema = [\n", + " bigquery.SchemaField(\"id\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"age\", \"INTEGER\", mode=\"NULLABLE\"), \n", + " bigquery.SchemaField(\"gender\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"height\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"weight\", \"FLOAT\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"ap_hi\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"ap_lo\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"cholesterol\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"gluc\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"smoke\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"alco\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"active\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"cardio\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"ageinyr\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"bmi\", \"FLOAT\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"bmicat\", \"STRING\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"agegroup\", \"STRING\", mode=\"NULLABLE\"),\n", + "\n", + "]\n", + "\n", + "table = bigquery.Table(table_id, schema=schema)\n", + "table = client.create_table(table) # Make an API request.\n", + "print(\n", + " \"Created table {}.{}.{}\".format(table.project, table.dataset_id, table.table_id)\n", + ")" + ] + }, + { + "cell_type": "markdown", + "id": "6e97b4c6-429c-4d36-a4c1-3963b079af6a", + "metadata": {}, + "source": [ + "This step is optional. We can make an API request to verify whether the dataset has been successfully created under our project." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "72f11cfd-8b20-48bf-b157-aaf791cd3a9f", + "metadata": {}, + "outputs": [], + "source": [ + "datasets = list(client.list_datasets()) # Make an API request.\n", + "project = project_id\n", + "\n", + "if datasets:\n", + " print(\"Datasets in project {}:\".format(project))\n", + " for dataset in datasets:\n", + " print(\"\\t{}\".format(dataset.dataset_id))\n", + "else:\n", + " print(\"{} project does not contain any datasets.\".format(project))" + ] + }, + { + "cell_type": "markdown", + "id": "ad42d610-daa6-4d92-ac14-2d90349dafe1", + "metadata": {}, + "source": [ + "We need to use the \"to_gbq\" function from the pandas-gbq library, which allows us to write a Pandas DataFrame to a Google BigQuery table. This enables us to populate our BigQuery table with the data from the DataFrame." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "7cfd1d66-a0fb-403d-a9b1-27250cc81f91", + "metadata": {}, + "outputs": [], + "source": [ + "df.to_gbq(table_id, project_id=project_id)\n", + "\n", + "from google.cloud import storage\n", + "import pandas as pd\n", + "from io import StringIO\n", + "\n", + "client.load_table_from_dataframe(df, table_id).result()" + ] + }, + { + "cell_type": "markdown", + "id": "4b16a292-afdc-415c-a8a7-071bc90281c3", + "metadata": {}, + "source": [ + "![image.png](../../NIHCloudLabGCP/images/gcp_rag_structure_data_01.png)" + ] + }, + { + "cell_type": "markdown", + "id": "22b4dd65-9bd9-48ef-9c98-85e3e9a82701", + "metadata": {}, + "source": [ + "This step is also optional. We can execute a simple query on the BigQuery table to count the number of records." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "29e91fc1-735f-4d6c-9fcf-3a2d745363c1", + "metadata": {}, + "outputs": [], + "source": [ + "query = \"\"\"\n", + " SELECT count(gender) as cgender\n", + " FROM `project_id.dataset_id.table_id`\n", + "\"\"\"\n", + "\n", + "# Execute the query\n", + "query_job = client.query(query)\n", + "\n", + "# Fetch the results\n", + "results = query_job.result()\n", + "\n", + "# Iterate through the rows\n", + "for row in results:\n", + " #print(f\"gender: {row.name}, age: {row.age}\")\n", + " print(f\"cgender: {row.cgender}\")\n", + "\n", + "try:\n", + " query_job = client.query(query)\n", + " results = query_job.result()\n", + " print(results)\n", + "except Exception as e:\n", + " print(f\"Error executing query: {str(e)}\")" + ] + }, + { + "cell_type": "markdown", + "id": "ce65ead3-694d-4d46-9376-38d35b8d0852", + "metadata": {}, + "source": [ + "cgender: 139920\n", + "" + ] + }, + { + "cell_type": "markdown", + "id": "b6fb53b8-9f93-4f37-8730-33c5871720fc", + "metadata": {}, + "source": [ + "To interact with the BigQuery table using a Pythonic domain language, we utilize SQLAlchemy. SQLAlchemy is a Python SQL toolkit that enables developers to access and manage SQL databases, allowing users to write queries as strings or chain Python objects for similar queries. " + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "d1d29b90-7bcd-46b8-82b9-dc5e58f9edee", + "metadata": {}, + "outputs": [], + "source": [ + "from google.cloud import bigquery\n", + "from sqlalchemy import *\n", + "from sqlalchemy.engine import create_engine\n", + "from sqlalchemy.schema import *\n", + "import os\n", + "from langchain_community.agent_toolkits import create_sql_agent\n", + "from langchain.agents import create_sql_agent\n", + "from langchain.agents.agent_toolkits import SQLDatabaseToolkit\n", + "from langchain.sql_database import SQLDatabase\n", + "\n", + "sqlalchemy_url = f'bigquery://{project_id}/{dataset}'\n", + "print(sqlalchemy_url)" + ] + }, + { + "cell_type": "markdown", + "id": "9194cc1d-7ef1-4c4e-8e65-e7d9ada5be24", + "metadata": {}, + "source": [ + "Next, we import the __ChatVertexAI__ agent from Langchain and configure it with the appropriate hyperparameters. In this instance, we are using the __gemini-1.5-pro__ LLM model, after which we create the SQL agent to enable querying the BigQuery table using a natural language string as a prompt. Temperature regulates randomness, with higher temperatures resulting in more varied and unpredictable outputs. Top-k sampling selects from the k most probable next tokens at each step, where a lower k emphasizes higher-probability tokens. The max tokens hyperparameter specifies the maximum number of tokens in the response from the large language model. Max retries indicates how many responses we will receive from the model." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "ee1899a2-dd79-4cf4-a02d-c7c35b99040b", + "metadata": {}, + "outputs": [], + "source": [ + "from langchain_google_vertexai import VertexAI, ChatVertexAI\n", + "\n", + "llm = ChatVertexAI(\n", + " model=\"gemini-1.5-pro\",\n", + " temperature=0,\n", + " max_tokens=8190,\n", + " timeout=None,\n", + " max_retries=2,\n", + ")\n", + "db = SQLDatabase.from_uri(sqlalchemy_url)\n", + "toolkit = SQLDatabaseToolkit(db=db, llm=llm)\n", + "agent_executor = create_sql_agent(\n", + " llm=llm,\n", + " toolkit=toolkit,\n", + " verbose=True,\n", + " top_k=100000\n", + " )\n" + ] + }, + { + "cell_type": "markdown", + "id": "f3551426-8644-48ba-b8c1-bf7d5eb843b3", + "metadata": {}, + "source": [ + "For our initial query, we check the number of rows to compare the result with what we obtained in a previous step using a simple SQL query against the BigQuery table. We can confirm that we received the same number of rows: 139,920." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "e9753533-e82b-47af-936a-a401d512308a", + "metadata": {}, + "outputs": [], + "source": [ + "agent_executor.run(\"How many rows are in the table screening? \")" + ] + }, + { + "cell_type": "markdown", + "id": "099154a1-34e2-4f6b-b146-cec6f5943cf7", + "metadata": {}, + "source": [ + "> Finished chain.\n", + "'The table screening has 139920 rows.'" + ] + }, + { + "cell_type": "markdown", + "id": "f2b62012-b2d2-48c1-ae69-6f8816065551", + "metadata": { + "tags": [] + }, + "source": [ + "Next, we pose a question that requires applying a filter, and we successfully obtain the accurate number of obese individuals in the table." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "5563d317-81ab-4a37-8225-e1b0dce20b36", + "metadata": {}, + "outputs": [], + "source": [ + "agent_executor.run(\"How many obese are in the table screen?\")" + ] + }, + { + "cell_type": "markdown", + "id": "40e1846d-1114-4ba1-b35a-65b47a838468", + "metadata": {}, + "source": [ + "> Finished chain.\n", + "'There are 37204 obese people in the table.'" + ] + }, + { + "cell_type": "markdown", + "id": "b96ea414-3f2d-4e66-ab0b-a034dfe507b0", + "metadata": {}, + "source": [ + "As a more complex query, we inquired about the number of female smokers, and the SQL agent accurately returned the answer of 1,626 female smokers." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "a456ff90-a3a5-4ee4-81da-fba4fcdf7f2d", + "metadata": {}, + "outputs": [], + "source": [ + "agent_executor.run(\"How many smokers are female in the table screen? the value 1 in smoke means that is a smoker and the value 2 in gender means that is a female\")" + ] + }, + { + "cell_type": "markdown", + "id": "7ea7810f-9701-4cb0-9430-181a57929884", + "metadata": {}, + "source": [ + "> Finished chain.\n", + "'There are 1626 female smokers in the table.'" + ] + }, + { + "cell_type": "markdown", + "id": "ca0fb657-19e0-4488-b9f6-25ca3971064f", + "metadata": { + "tags": [] + }, + "source": [ + "## Conclusion" + ] + }, + { + "cell_type": "markdown", + "id": "7f85e560-231f-4841-b1d0-334326e4b347", + "metadata": {}, + "source": [ + "You have learned how to create a dataset and a table in BigQuery, as well as how to set up an agent that enables you to query the table using natural language instead of SQL queries, allowing you to obtain the results you need." + ] + }, + { + "cell_type": "markdown", + "id": "41573388-7feb-44b6-9e8a-0511cc8ae62e", + "metadata": {}, + "source": [ + "## Clean Up" + ] + }, + { + "cell_type": "markdown", + "id": "1f0edc0b-f2cc-478c-855f-9acff3792f4d", + "metadata": {}, + "source": [ + "Please remember to delete or stop your Jupyter notebook and delete your BigQuery dataset and table to prevent incurring charges. And if you have created any other services like buckets, please remember to delete them as well." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "fe73dde2-056a-46a3-bacb-0a6671f7128f", + "metadata": {}, + "outputs": [], + "source": [] + } + ], + "metadata": { + "environment": { + "kernel": "python3", + "name": "common-cpu.m125", + "type": "gcloud", + "uri": "us-docker.pkg.dev/deeplearning-platform-release/gcr.io/base-cpu:m125" + }, + "kernelspec": { + "display_name": "Python 3 (Local)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.10.15" + } + }, + "nbformat": 4, + "nbformat_minor": 5 +} diff --git a/notebooks/GenAI/.ipynb_checkpoints/GCP_RAG_for_Structure_Data_old-checkpoint.ipynb b/notebooks/GenAI/.ipynb_checkpoints/GCP_RAG_for_Structure_Data_old-checkpoint.ipynb new file mode 100644 index 0000000..a7d082e --- /dev/null +++ b/notebooks/GenAI/.ipynb_checkpoints/GCP_RAG_for_Structure_Data_old-checkpoint.ipynb @@ -0,0 +1,710 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "id": "6e38b3f2-2b29-4788-a29a-f9e5d10f8365", + "metadata": {}, + "source": [ + "##### __Skill Level:__ Intermediate" + ] + }, + { + "cell_type": "markdown", + "id": "6c0533e3-4fbd-47b1-8458-fd193bcb263f", + "metadata": {}, + "source": [ + "# Creating a Chatbot for Structure Data using a RAG" + ] + }, + { + "cell_type": "markdown", + "id": "76cf4b02-6422-4a48-882c-c00315e34c8a", + "metadata": { + "tags": [] + }, + "source": [ + "## Overview" + ] + }, + { + "cell_type": "markdown", + "id": "7b7870dd-09e0-4f6a-91a9-7c30529af720", + "metadata": {}, + "source": [ + "Generative AI (GenAI) represents a transformative technology capable of producing human-like text, images, code, and various other types of content. While much of the focus has been on unstructured data—such as PDFs, text documents, image files, and websites—many GenAI implementations rely on a parameter called \"top K.\" This algorithm retrieves only the highest-scoring pieces of content relevant to a user's query, which can be limiting. Users seeking insights from structured data formats like CSV and JSON often require access to all relevant occurrences, rather than just a subset.\n", + "\n", + "In this tutorial, we present a technique that leverages SQL databases. By formulating a query based on the user's request, the model submits this query to the database, providing comprehensive results. This approach not only ensures users receive all pertinent information but also reduces the likelihood of exceeding token limits." + ] + }, + { + "cell_type": "markdown", + "id": "d513a9fe-1ac4-4610-b53c-6696e461f828", + "metadata": {}, + "source": [ + "## Prerequisites" + ] + }, + { + "cell_type": "markdown", + "id": "fbdb09aa-73d6-4cee-b3e7-5d6b243e3d58", + "metadata": {}, + "source": [ + "We assume you have access to Vertex AI and have enabled the necessary APIs.\n", + "\n", + "In this tutorial, we will be using Google Gemini Pro 1.5, which does not require deployment. However, if you prefer to use a different model, you can select one from the Model Garden via the console. This will allow you to add the model to your registry, create an endpoint (or utilize an existing one), and deploy the model—all in a single step. Here is a link for more information: [model deployment](https://cloud.google.com/vertex-ai/docs/general/deployment).\n", + "\n", + "Before we begin, you'll need to create a Vertex AI RAG Data Service Agent service account. To do this, go to the IAM section of the console. Ensure you check the box for \"Include Google-provided role grant.\" If the role is not listed, click \"Grant Access\" and add \"Vertex AI RAG Data Service Agent\" as a role." + ] + }, + { + "cell_type": "markdown", + "id": "eeb24716-0db8-462f-8faa-0b8ad9d61feb", + "metadata": { + "tags": [] + }, + "source": [ + "## Learning objectives" + ] + }, + { + "cell_type": "markdown", + "id": "f5216a43-31ee-4c3e-821a-4b81f4be8e54", + "metadata": {}, + "source": [ + "In this tutorial you will learn about:\n", + "- How to set up a BigQuery dataset and a table.\n", + "- How to load data to a BigQuery table.\n", + "- How to use the langchain ChatVertexAI agent to extract information froim the table. \n" + ] + }, + { + "cell_type": "markdown", + "id": "348910aa-9802-4318-8fcb-86603158923b", + "metadata": { + "tags": [] + }, + "source": [ + "## Pricing" + ] + }, + { + "cell_type": "markdown", + "id": "0dca31b2-880d-4cbc-bf76-ede96998e371", + "metadata": {}, + "source": [ + "
    \n", + "
  • BigQuery Standard 1TB Storage.
  • \n", + "
  • Model for Chat: 5 request per day. Average Input and Output characters 5,000.
  • \n", + "
  • Total $23.89 per month.
  • \n", + "
" + ] + }, + { + "cell_type": "markdown", + "id": "c7d9e3ee-1c7e-4baa-a818-c13330c665dd", + "metadata": {}, + "source": [ + "## Get Start" + ] + }, + { + "cell_type": "markdown", + "id": "137defcd-fd2b-421d-9b2e-fe2b878fb325", + "metadata": {}, + "source": [ + "### Install Packages" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "55c14a38-ba51-466d-90b5-1512f5a7fae0", + "metadata": { + "tags": [] + }, + "outputs": [], + "source": [ + "!pip install google-cloud-bigquery\n", + "!pip install gcloud\n", + "!pip install langchain\n", + "!pip install -U langchain-community\n", + "!pip install -qU langchain_google_vertexai\n", + "!pip install pandas-gbq" + ] + }, + { + "cell_type": "markdown", + "id": "64e22773-9a03-47de-9b6d-7946abcba722", + "metadata": {}, + "source": [ + "Set your project id, location, and bucket variables." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "7193a9ac-3169-45c1-9c5d-ba5e2f31458f", + "metadata": {}, + "outputs": [], + "source": [ + "project_id=''\n", + "location=' (e.g.us-east4)'\n", + "bucket = ''" + ] + }, + { + "cell_type": "markdown", + "id": "df0d9eda-ee54-4c4c-86e6-9166b2b82e2d", + "metadata": {}, + "source": [ + "Create a bucket where the data used for tutorial is going to be placed." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "7b583234-2d92-4e5b-841d-c7d79dd87d45", + "metadata": {}, + "outputs": [], + "source": [ + "# make bucket\n", + "!gsutil mb -l {location} gs://{bucket}" + ] + }, + { + "cell_type": "markdown", + "id": "5781a07c-84ed-416c-a782-b1dc881e5c18", + "metadata": {}, + "source": [ + "Provide the names of the bucket name, source file name path, and destination blob name to upload the CSV source file to the bucket." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "4829094e-61a6-4dd7-854f-6b21dc19b1b7", + "metadata": {}, + "outputs": [], + "source": [ + "from google.cloud import storage\n", + "\n", + "# The ID of your GCS bucket\n", + "bucket_name = bucket\n", + " # The path to your file to upload\n", + "source_file_name = \"\"\n", + " # The ID of your GCS object\n", + "destination_blob_name = \"\"\n", + "\n", + "storage_client = storage.Client()\n", + "bucket = storage_client.bucket(bucket_name)\n", + "blob = bucket.blob(destination_blob_name)\n", + "blob.upload_from_filename(source_file_name)\n", + "\n", + "print(\n", + " \"File {} uploaded to {}.\".format(\n", + " source_file_name, destination_blob_name\n", + " )\n", + ")\n" + ] + }, + { + "cell_type": "markdown", + "id": "7a00dc83-8713-4ea5-9345-08a9f3d785d0", + "metadata": {}, + "source": [ + "The next step is to create a database in BigQuery. To accomplish this, we need to define a dataset_id and construct a dataset object that will be sent to the API for creation. Note that in BigQuery, a dataset is analogous to a database." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "1750e13d-ec05-4335-82a2-ffe54c5b5f32", + "metadata": {}, + "outputs": [], + "source": [ + "from google.cloud import bigquery\n", + "\n", + "client = bigquery.Client()\n", + "\n", + "# Set dataset_id to the ID of the dataset to create.\n", + "dataset_id = project_id.''\n", + "\n", + "# Construct a full Dataset object to send to the API.\n", + "dataset = bigquery.Dataset(dataset_id)\n", + "\n", + "# Specify the geographic location where the dataset should reside.\n", + "dataset.location = location\n", + "\n", + "# Send the dataset to the API for creation, with an explicit timeout.\n", + "# Raises google.api_core.exceptions.Conflict if the Dataset already\n", + "# exists within the project.\n", + "dataset = client.create_dataset(dataset, timeout=30) # Make an API request.\n", + "print(\"Created dataset {}.{}\".format(client.project, dataset.dataset_id))" + ] + }, + { + "cell_type": "markdown", + "id": "544c191c-2bc0-4a47-a0a7-cea81035597a", + "metadata": {}, + "source": [ + "Once the dataset is created, the next step is to create a table. First, we define a table ID, followed by outlining the table's schema. Finally, we send an API request to create the table within the dataset established in the previous step." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "68206b14-1b5e-49c3-9f07-52ba6c70a93e", + "metadata": {}, + "outputs": [], + "source": [ + "# Set table_id to the ID of the table to create.\n", + "\n", + "client = bigquery.Client()\n", + "\n", + "table_id = project_id.dataset_id.\"
\"\n", + "\n", + "schema = [\n", + " bigquery.SchemaField(\"id\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"age\", \"INTEGER\", mode=\"NULLABLE\"), \n", + " bigquery.SchemaField(\"gender\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"height\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"weight\", \"FLOAT\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"ap_hi\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"ap_lo\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"cholesterol\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"gluc\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"smoke\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"alco\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"active\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"cardio\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"ageinyr\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"bmi\", \"FLOAT\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"bmicat\", \"STRING\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"agegroup\", \"STRING\", mode=\"NULLABLE\"),\n", + "\n", + "]\n", + "\n", + "table = bigquery.Table(table_id, schema=schema)\n", + "table = client.create_table(table) # Make an API request.\n", + "print(\n", + " \"Created table {}.{}.{}\".format(table.project, table.dataset_id, table.table_id)\n", + ")" + ] + }, + { + "cell_type": "markdown", + "id": "6e97b4c6-429c-4d36-a4c1-3963b079af6a", + "metadata": {}, + "source": [ + "This step is optional. We can make an API request to verify whether the dataset has been successfully created under our project." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "72f11cfd-8b20-48bf-b157-aaf791cd3a9f", + "metadata": {}, + "outputs": [], + "source": [ + "datasets = list(client.list_datasets()) # Make an API request.\n", + "project = project_id\n", + "\n", + "if datasets:\n", + " print(\"Datasets in project {}:\".format(project))\n", + " for dataset in datasets:\n", + " print(\"\\t{}\".format(dataset.dataset_id))\n", + "else:\n", + " print(\"{} project does not contain any datasets.\".format(project))" + ] + }, + { + "cell_type": "markdown", + "id": "dad4f96d-ae52-43f2-8290-c04b25960ea8", + "metadata": {}, + "source": [ + "We can perform a similar check to confirm that the table we created is located within the appropriate dataset." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "80ea826c-9d70-4268-b4db-a657d5cdcaf0", + "metadata": {}, + "outputs": [], + "source": [ + "# Set dataset_id to the ID of the dataset to fetch.\n", + "\n", + "dataset = client.get_dataset(dataset_id) # Make an API request.\n", + "\n", + "full_dataset_id = \"{}.{}\".format(dataset.project, dataset.dataset_id)\n", + "friendly_name = dataset.friendly_name\n", + "print(\n", + " \"Got dataset '{}' with friendly_name '{}'.\".format(\n", + " full_dataset_id, friendly_name\n", + " )\n", + ")\n", + "\n", + "# View dataset properties.\n", + "print(\"Description: {}\".format(dataset.description))\n", + "print(\"Labels:\")\n", + "labels = dataset.labels\n", + "if labels:\n", + " for label, value in labels.items():\n", + " print(\"\\t{}: {}\".format(label, value))\n", + "else:\n", + " print(\"\\tDataset has no labels defined.\")\n", + "\n", + "# View tables in dataset.\n", + "print(\"Tables:\")\n", + "tables = list(client.list_tables(dataset)) # Make an API request(s).\n", + "if tables:\n", + " for table in tables:\n", + " print(\"\\t{}\".format(table.table_id))\n", + "else:\n", + " print(\"\\tThis dataset does not contain any tables.\")" + ] + }, + { + "cell_type": "markdown", + "id": "736080b9-0cd9-44fd-9580-dadb10a22a59", + "metadata": { + "tags": [] + }, + "source": [ + "To upload the data from our file stored in the bucket, we first need to create a DataFrame. In this example, we will be using a file that contains health screening data." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "b614cc1a-b318-4364-8d47-1dea9700c792", + "metadata": {}, + "outputs": [], + "source": [ + "from google.cloud import storage\n", + "import pandas as pd\n", + "from io import StringIO\n", + "\n", + "# Initialize a client\n", + "client = storage.Client()\n", + "my_bucket = bucket\n", + "storage_client = storage.Client()\n", + "bucket = storage_client.get_bucket(my_bucket)\n", + "blob = bucket.blob(destination_blob_name)\n", + "path = \"\"\n", + "df = pd.read_csv(path)\n", + "df.head()" + ] + }, + { + "cell_type": "markdown", + "id": "ad42d610-daa6-4d92-ac14-2d90349dafe1", + "metadata": {}, + "source": [ + "We need to use the \"to_gbq\" function from the pandas-gbq library, which allows us to write a Pandas DataFrame to a Google BigQuery table. This enables us to populate our BigQuery table with the data from the DataFrame." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "7cfd1d66-a0fb-403d-a9b1-27250cc81f91", + "metadata": {}, + "outputs": [], + "source": [ + "df.to_gbq(table_id, project_id=project_id)\n", + "\n", + "from google.cloud import storage\n", + "import pandas as pd\n", + "from io import StringIO\n", + "\n", + "client.load_table_from_dataframe(df, table_id).result()" + ] + }, + { + "cell_type": "markdown", + "id": "22b4dd65-9bd9-48ef-9c98-85e3e9a82701", + "metadata": {}, + "source": [ + "This step is also optional. We can execute a simple query on the BigQuery table to count the number of records." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "29e91fc1-735f-4d6c-9fcf-3a2d745363c1", + "metadata": {}, + "outputs": [], + "source": [ + "query = \"\"\"\n", + " SELECT count(gender) as cgender\n", + " FROM `project_id.dataset_id.table_id`\n", + "\"\"\"\n", + "\n", + "# Execute the query\n", + "query_job = client.query(query)\n", + "\n", + "# Fetch the results\n", + "results = query_job.result()\n", + "\n", + "# Iterate through the rows\n", + "for row in results:\n", + " #print(f\"gender: {row.name}, age: {row.age}\")\n", + " print(f\"cgender: {row.cgender}\")\n", + "\n", + "try:\n", + " query_job = client.query(query)\n", + " results = query_job.result()\n", + " print(results)\n", + "except Exception as e:\n", + " print(f\"Error executing query: {str(e)}\")" + ] + }, + { + "cell_type": "markdown", + "id": "ce65ead3-694d-4d46-9376-38d35b8d0852", + "metadata": {}, + "source": [ + "cgender: 139920\n", + "" + ] + }, + { + "cell_type": "markdown", + "id": "b6fb53b8-9f93-4f37-8730-33c5871720fc", + "metadata": {}, + "source": [ + "To interact with the BigQuery table using a Pythonic domain language, we utilize SQLAlchemy. SQLAlchemy is a Python SQL toolkit that enables developers to access and manage SQL databases, allowing users to write queries as strings or chain Python objects for similar queries. However, to do this, we need specific credentials, which can be accessed through a service account file. For more information on how to create a service account file, please visit the following link: [Service Account Creation](https://cloud.google.com/iam/docs/service-accounts-create#python)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "d1d29b90-7bcd-46b8-82b9-dc5e58f9edee", + "metadata": {}, + "outputs": [], + "source": [ + "from google.cloud import bigquery\n", + "from sqlalchemy import *\n", + "from sqlalchemy.engine import create_engine\n", + "from sqlalchemy.schema import *\n", + "import os\n", + "from langchain_community.agent_toolkits import create_sql_agent\n", + "from langchain.agents import create_sql_agent\n", + "from langchain.agents.agent_toolkits import SQLDatabaseToolkit\n", + "from langchain.sql_database import SQLDatabase\n", + "service_account_file = \"\" # Change to where your service account key file is located\n", + "project_id = project_id\n", + "dataset = \"\"\n", + "table = \"
\"\n", + "sqlalchemy_url = f'bigquery://{project_id}/{dataset}?credentials_path={service_account_file}'\n", + "print(sqlalchemy_url)" + ] + }, + { + "cell_type": "markdown", + "id": "9194cc1d-7ef1-4c4e-8e65-e7d9ada5be24", + "metadata": {}, + "source": [ + "Next, we import the __ChatVertexAI__ agent from Langchain and configure it with the appropriate hyperparameters. In this instance, we are using the __gemini-1.5-pro__ LLM model, after which we create the SQL agent to enable querying the BigQuery table using a natural language string as a prompt." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "ee1899a2-dd79-4cf4-a02d-c7c35b99040b", + "metadata": {}, + "outputs": [], + "source": [ + "from langchain_google_vertexai import VertexAI, ChatVertexAI\n", + "\n", + "llm = ChatVertexAI(\n", + " model=\"gemini-1.5-pro\",\n", + " temperature=0,\n", + " max_tokens=8190,\n", + " timeout=None,\n", + " max_retries=2,\n", + ")\n", + "db = SQLDatabase.from_uri(sqlalchemy_url)\n", + "toolkit = SQLDatabaseToolkit(db=db, llm=llm)\n", + "agent_executor = create_sql_agent(\n", + " llm=llm,\n", + " toolkit=toolkit,\n", + " verbose=True,\n", + " top_k=100000\n", + " )\n" + ] + }, + { + "cell_type": "markdown", + "id": "f3551426-8644-48ba-b8c1-bf7d5eb843b3", + "metadata": {}, + "source": [ + "For our initial query, we check the number of rows to compare the result with what we obtained in a previous step using a simple SQL query against the BigQuery table. We can confirm that we received the same number of rows: 139,920." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "e9753533-e82b-47af-936a-a401d512308a", + "metadata": {}, + "outputs": [], + "source": [ + "agent_executor.run(\"How many rows are in the table screening? \")" + ] + }, + { + "cell_type": "markdown", + "id": "099154a1-34e2-4f6b-b146-cec6f5943cf7", + "metadata": {}, + "source": [ + "> Finished chain.\n", + "'The table screening has 139920 rows.'" + ] + }, + { + "cell_type": "markdown", + "id": "65f645ed-8efa-436f-b993-72c45a7aadbe", + "metadata": { + "tags": [] + }, + "source": [ + "We know the output of the model are correct. The image below is a reminder of what the actual results are calculated from." + ] + }, + { + "cell_type": "markdown", + "id": "8160ee6e-f95f-44c3-ab27-4415b44ba461", + "metadata": { + "tags": [] + }, + "source": [ + "![image.png](../../images/gcp_rag_structure_data_01.png)" + ] + }, + { + "cell_type": "markdown", + "id": "f2b62012-b2d2-48c1-ae69-6f8816065551", + "metadata": { + "tags": [] + }, + "source": [ + "Next, we pose a question that requires applying a filter, and we successfully obtain the accurate number of obese individuals in the table." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "5563d317-81ab-4a37-8225-e1b0dce20b36", + "metadata": {}, + "outputs": [], + "source": [ + "agent_executor.run(\"How many obese are in the table screen?\")" + ] + }, + { + "cell_type": "markdown", + "id": "40e1846d-1114-4ba1-b35a-65b47a838468", + "metadata": {}, + "source": [ + "> Finished chain.\n", + "'There are 37204 obese people in the table.'" + ] + }, + { + "cell_type": "markdown", + "id": "b96ea414-3f2d-4e66-ab0b-a034dfe507b0", + "metadata": {}, + "source": [ + "As a more complex query, we inquired about the number of female smokers, and the SQL agent accurately returned the answer of 1,626 female smokers." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "a456ff90-a3a5-4ee4-81da-fba4fcdf7f2d", + "metadata": {}, + "outputs": [], + "source": [ + "agent_executor.run(\"How many smokers are female in the table screen? the value 1 in smoke means that is a smoker and the value 2 in gender means that is a female\")" + ] + }, + { + "cell_type": "markdown", + "id": "7ea7810f-9701-4cb0-9430-181a57929884", + "metadata": {}, + "source": [ + "> Finished chain.\n", + "'There are 1626 female smokers in the table.'" + ] + }, + { + "cell_type": "markdown", + "id": "ca0fb657-19e0-4488-b9f6-25ca3971064f", + "metadata": { + "tags": [] + }, + "source": [ + "## Conclusion" + ] + }, + { + "cell_type": "markdown", + "id": "7f85e560-231f-4841-b1d0-334326e4b347", + "metadata": {}, + "source": [ + "You have learned how to create a dataset and a table in BigQuery, as well as how to set up an agent that enables you to query the table using natural language instead of SQL queries, allowing you to obtain the results you need." + ] + }, + { + "cell_type": "markdown", + "id": "41573388-7feb-44b6-9e8a-0511cc8ae62e", + "metadata": {}, + "source": [ + "## Clean Up" + ] + }, + { + "cell_type": "markdown", + "id": "1f0edc0b-f2cc-478c-855f-9acff3792f4d", + "metadata": {}, + "source": [ + "Please remember to delete or stop your Jupyter notebook and delete your data store to prevent incurring charges. And if you have created any other services like buckets, please remember to delete them as well." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "fe73dde2-056a-46a3-bacb-0a6671f7128f", + "metadata": {}, + "outputs": [], + "source": [] + } + ], + "metadata": { + "environment": { + "kernel": "python3", + "name": "common-cpu.m125", + "type": "gcloud", + "uri": "us-docker.pkg.dev/deeplearning-platform-release/gcr.io/base-cpu:m125" + }, + "kernelspec": { + "display_name": "Python 3 (Local)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.10.15" + } + }, + "nbformat": 4, + "nbformat_minor": 5 +} diff --git a/notebooks/GenAI/GCP_RAG_for_Structure_Data.ipynb b/notebooks/GenAI/GCP_RAG_for_Structure_Data.ipynb index a7d082e..54d4614 100644 --- a/notebooks/GenAI/GCP_RAG_for_Structure_Data.ipynb +++ b/notebooks/GenAI/GCP_RAG_for_Structure_Data.ipynb @@ -33,7 +33,7 @@ "source": [ "Generative AI (GenAI) represents a transformative technology capable of producing human-like text, images, code, and various other types of content. While much of the focus has been on unstructured data—such as PDFs, text documents, image files, and websites—many GenAI implementations rely on a parameter called \"top K.\" This algorithm retrieves only the highest-scoring pieces of content relevant to a user's query, which can be limiting. Users seeking insights from structured data formats like CSV and JSON often require access to all relevant occurrences, rather than just a subset.\n", "\n", - "In this tutorial, we present a technique that leverages SQL databases. By formulating a query based on the user's request, the model submits this query to the database, providing comprehensive results. This approach not only ensures users receive all pertinent information but also reduces the likelihood of exceeding token limits." + "In this tutorial, we use a RAG agent together with a VertexAI chat model gemini-1.5-pro to query the BigQuery table. A Retrieval Augmented Generation (RAG) agent is a key part of a RAG application that enhances the capabilities of the large language models (LLMs) by integrating external data retrieval. AI agents empower LLMs to interact with the world through actions and tools." ] }, { @@ -53,7 +53,9 @@ "\n", "In this tutorial, we will be using Google Gemini Pro 1.5, which does not require deployment. However, if you prefer to use a different model, you can select one from the Model Garden via the console. This will allow you to add the model to your registry, create an endpoint (or utilize an existing one), and deploy the model—all in a single step. Here is a link for more information: [model deployment](https://cloud.google.com/vertex-ai/docs/general/deployment).\n", "\n", - "Before we begin, you'll need to create a Vertex AI RAG Data Service Agent service account. To do this, go to the IAM section of the console. Ensure you check the box for \"Include Google-provided role grant.\" If the role is not listed, click \"Grant Access\" and add \"Vertex AI RAG Data Service Agent\" as a role." + "Before we begin, you'll need to create a Vertex AI RAG Data Service Agent service account. To do this, go to the IAM section of the console. Ensure you check the box for \"Include Google-provided role grant.\" If the role is not listed, click \"Grant Access\" and add \"Vertex AI RAG Data Service Agent\" as a role.\n", + "\n", + "We are using a e2-standard-4 (Efficient Instance: 4 vCPUs, 16 GB RAM) for this tutorial. " ] }, { @@ -74,7 +76,7 @@ "In this tutorial you will learn about:\n", "- How to set up a BigQuery dataset and a table.\n", "- How to load data to a BigQuery table.\n", - "- How to use the langchain ChatVertexAI agent to extract information froim the table. \n" + "- How to use the langchain ChatVertexAI agent to extract information from the table. \n" ] }, { @@ -92,11 +94,7 @@ "id": "0dca31b2-880d-4cbc-bf76-ede96998e371", "metadata": {}, "source": [ - "
    \n", - "
  • BigQuery Standard 1TB Storage.
  • \n", - "
  • Model for Chat: 5 request per day. Average Input and Output characters 5,000.
  • \n", - "
  • Total $23.89 per month.
  • \n", - "
" + "If you are following this tutorial in one sitting it will cost $23.89 per month. Completing the process in multiple sessions or using a method different from the tutorial may result in increased costs." ] }, { @@ -176,7 +174,7 @@ "id": "5781a07c-84ed-416c-a782-b1dc881e5c18", "metadata": {}, "source": [ - "Provide the names of the bucket name, source file name path, and destination blob name to upload the CSV source file to the bucket." + "Once the bucket is created, we need to access the CSV source file. In this tutorial, I transferred the data file to our Jupyter notebook by simply dragging and dropping it from my local folder. Next, we need to specify the bucket name and the path of the data source in order to upload the CSV file to the bucket. It is important to keep in mind that the name of the bucket has to be unique." ] }, { @@ -186,25 +184,7 @@ "metadata": {}, "outputs": [], "source": [ - "from google.cloud import storage\n", - "\n", - "# The ID of your GCS bucket\n", - "bucket_name = bucket\n", - " # The path to your file to upload\n", - "source_file_name = \"\"\n", - " # The ID of your GCS object\n", - "destination_blob_name = \"\"\n", - "\n", - "storage_client = storage.Client()\n", - "bucket = storage_client.bucket(bucket_name)\n", - "blob = bucket.blob(destination_blob_name)\n", - "blob.upload_from_filename(source_file_name)\n", - "\n", - "print(\n", - " \"File {} uploaded to {}.\".format(\n", - " source_file_name, destination_blob_name\n", - " )\n", - ")\n" + "!gsutil cp '' gs://{bucket}" ] }, { @@ -226,8 +206,9 @@ "\n", "client = bigquery.Client()\n", "\n", + "dataset_name = ''\n", "# Set dataset_id to the ID of the dataset to create.\n", - "dataset_id = project_id.''\n", + "dataset_id = f'{project_id}.{dataset_name}'\n", "\n", "# Construct a full Dataset object to send to the API.\n", "dataset = bigquery.Dataset(dataset_id)\n", @@ -261,7 +242,8 @@ "\n", "client = bigquery.Client()\n", "\n", - "table_id = project_id.dataset_id.\"
\"\n", + "table_name = \"
\"\n", + "table_id = f'{project_id}.{dataset_id}.{table_name}'\n", "\n", "schema = [\n", " bigquery.SchemaField(\"id\", \"INTEGER\", mode=\"NULLABLE\"),\n", @@ -319,105 +301,34 @@ }, { "cell_type": "markdown", - "id": "dad4f96d-ae52-43f2-8290-c04b25960ea8", + "id": "ad42d610-daa6-4d92-ac14-2d90349dafe1", "metadata": {}, "source": [ - "We can perform a similar check to confirm that the table we created is located within the appropriate dataset." + "We need to use the \"to_gbq\" function from the pandas-gbq library, which allows us to write a Pandas DataFrame to a Google BigQuery table. This enables us to populate our BigQuery table with the data from the DataFrame." ] }, { "cell_type": "code", "execution_count": null, - "id": "80ea826c-9d70-4268-b4db-a657d5cdcaf0", + "id": "7cfd1d66-a0fb-403d-a9b1-27250cc81f91", "metadata": {}, "outputs": [], "source": [ - "# Set dataset_id to the ID of the dataset to fetch.\n", - "\n", - "dataset = client.get_dataset(dataset_id) # Make an API request.\n", - "\n", - "full_dataset_id = \"{}.{}\".format(dataset.project, dataset.dataset_id)\n", - "friendly_name = dataset.friendly_name\n", - "print(\n", - " \"Got dataset '{}' with friendly_name '{}'.\".format(\n", - " full_dataset_id, friendly_name\n", - " )\n", - ")\n", - "\n", - "# View dataset properties.\n", - "print(\"Description: {}\".format(dataset.description))\n", - "print(\"Labels:\")\n", - "labels = dataset.labels\n", - "if labels:\n", - " for label, value in labels.items():\n", - " print(\"\\t{}: {}\".format(label, value))\n", - "else:\n", - " print(\"\\tDataset has no labels defined.\")\n", + "df.to_gbq(table_id, project_id=project_id)\n", "\n", - "# View tables in dataset.\n", - "print(\"Tables:\")\n", - "tables = list(client.list_tables(dataset)) # Make an API request(s).\n", - "if tables:\n", - " for table in tables:\n", - " print(\"\\t{}\".format(table.table_id))\n", - "else:\n", - " print(\"\\tThis dataset does not contain any tables.\")" - ] - }, - { - "cell_type": "markdown", - "id": "736080b9-0cd9-44fd-9580-dadb10a22a59", - "metadata": { - "tags": [] - }, - "source": [ - "To upload the data from our file stored in the bucket, we first need to create a DataFrame. In this example, we will be using a file that contains health screening data." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "id": "b614cc1a-b318-4364-8d47-1dea9700c792", - "metadata": {}, - "outputs": [], - "source": [ "from google.cloud import storage\n", "import pandas as pd\n", "from io import StringIO\n", "\n", - "# Initialize a client\n", - "client = storage.Client()\n", - "my_bucket = bucket\n", - "storage_client = storage.Client()\n", - "bucket = storage_client.get_bucket(my_bucket)\n", - "blob = bucket.blob(destination_blob_name)\n", - "path = \"\"\n", - "df = pd.read_csv(path)\n", - "df.head()" + "client.load_table_from_dataframe(df, table_id).result()" ] }, { "cell_type": "markdown", - "id": "ad42d610-daa6-4d92-ac14-2d90349dafe1", + "id": "4b16a292-afdc-415c-a8a7-071bc90281c3", "metadata": {}, "source": [ - "We need to use the \"to_gbq\" function from the pandas-gbq library, which allows us to write a Pandas DataFrame to a Google BigQuery table. This enables us to populate our BigQuery table with the data from the DataFrame." - ] - }, - { - "cell_type": "code", - "execution_count": null, - "id": "7cfd1d66-a0fb-403d-a9b1-27250cc81f91", - "metadata": {}, - "outputs": [], - "source": [ - "df.to_gbq(table_id, project_id=project_id)\n", - "\n", - "from google.cloud import storage\n", - "import pandas as pd\n", - "from io import StringIO\n", - "\n", - "client.load_table_from_dataframe(df, table_id).result()" + "![image.png](../../NIHCloudLabGCP/images/gcp_rag_structure_data_01.png)" ] }, { @@ -473,7 +384,7 @@ "id": "b6fb53b8-9f93-4f37-8730-33c5871720fc", "metadata": {}, "source": [ - "To interact with the BigQuery table using a Pythonic domain language, we utilize SQLAlchemy. SQLAlchemy is a Python SQL toolkit that enables developers to access and manage SQL databases, allowing users to write queries as strings or chain Python objects for similar queries. However, to do this, we need specific credentials, which can be accessed through a service account file. For more information on how to create a service account file, please visit the following link: [Service Account Creation](https://cloud.google.com/iam/docs/service-accounts-create#python)" + "To interact with the BigQuery table using a Pythonic domain language, we utilize SQLAlchemy. SQLAlchemy is a Python SQL toolkit that enables developers to access and manage SQL databases, allowing users to write queries as strings or chain Python objects for similar queries. " ] }, { @@ -492,11 +403,8 @@ "from langchain.agents import create_sql_agent\n", "from langchain.agents.agent_toolkits import SQLDatabaseToolkit\n", "from langchain.sql_database import SQLDatabase\n", - "service_account_file = \"\" # Change to where your service account key file is located\n", - "project_id = project_id\n", - "dataset = \"\"\n", - "table = \"
\"\n", - "sqlalchemy_url = f'bigquery://{project_id}/{dataset}?credentials_path={service_account_file}'\n", + "\n", + "sqlalchemy_url = f'bigquery://{project_id}/{dataset}'\n", "print(sqlalchemy_url)" ] }, @@ -505,7 +413,7 @@ "id": "9194cc1d-7ef1-4c4e-8e65-e7d9ada5be24", "metadata": {}, "source": [ - "Next, we import the __ChatVertexAI__ agent from Langchain and configure it with the appropriate hyperparameters. In this instance, we are using the __gemini-1.5-pro__ LLM model, after which we create the SQL agent to enable querying the BigQuery table using a natural language string as a prompt." + "Next, we import the __ChatVertexAI__ agent from Langchain and configure it with the appropriate hyperparameters. In this instance, we are using the __gemini-1.5-pro__ LLM model, after which we create the SQL agent to enable querying the BigQuery table using a natural language string as a prompt. Temperature regulates randomness, with higher temperatures resulting in more varied and unpredictable outputs. Top-k sampling selects from the k most probable next tokens at each step, where a lower k emphasizes higher-probability tokens. The max tokens hyperparameter specifies the maximum number of tokens in the response from the large language model. Max retries indicates how many responses we will receive from the model." ] }, { @@ -561,26 +469,6 @@ "'The table screening has 139920 rows.'" ] }, - { - "cell_type": "markdown", - "id": "65f645ed-8efa-436f-b993-72c45a7aadbe", - "metadata": { - "tags": [] - }, - "source": [ - "We know the output of the model are correct. The image below is a reminder of what the actual results are calculated from." - ] - }, - { - "cell_type": "markdown", - "id": "8160ee6e-f95f-44c3-ab27-4415b44ba461", - "metadata": { - "tags": [] - }, - "source": [ - "![image.png](../../images/gcp_rag_structure_data_01.png)" - ] - }, { "cell_type": "markdown", "id": "f2b62012-b2d2-48c1-ae69-6f8816065551", @@ -668,7 +556,7 @@ "id": "1f0edc0b-f2cc-478c-855f-9acff3792f4d", "metadata": {}, "source": [ - "Please remember to delete or stop your Jupyter notebook and delete your data store to prevent incurring charges. And if you have created any other services like buckets, please remember to delete them as well." + "Please remember to delete or stop your Jupyter notebook and delete your BigQuery dataset and table to prevent incurring charges. And if you have created any other services like buckets, please remember to delete them as well." ] }, { diff --git a/notebooks/GenAI/GCP_RAG_for_Structure_Data_old.ipynb b/notebooks/GenAI/GCP_RAG_for_Structure_Data_old.ipynb new file mode 100644 index 0000000..a7d082e --- /dev/null +++ b/notebooks/GenAI/GCP_RAG_for_Structure_Data_old.ipynb @@ -0,0 +1,710 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "id": "6e38b3f2-2b29-4788-a29a-f9e5d10f8365", + "metadata": {}, + "source": [ + "##### __Skill Level:__ Intermediate" + ] + }, + { + "cell_type": "markdown", + "id": "6c0533e3-4fbd-47b1-8458-fd193bcb263f", + "metadata": {}, + "source": [ + "# Creating a Chatbot for Structure Data using a RAG" + ] + }, + { + "cell_type": "markdown", + "id": "76cf4b02-6422-4a48-882c-c00315e34c8a", + "metadata": { + "tags": [] + }, + "source": [ + "## Overview" + ] + }, + { + "cell_type": "markdown", + "id": "7b7870dd-09e0-4f6a-91a9-7c30529af720", + "metadata": {}, + "source": [ + "Generative AI (GenAI) represents a transformative technology capable of producing human-like text, images, code, and various other types of content. While much of the focus has been on unstructured data—such as PDFs, text documents, image files, and websites—many GenAI implementations rely on a parameter called \"top K.\" This algorithm retrieves only the highest-scoring pieces of content relevant to a user's query, which can be limiting. Users seeking insights from structured data formats like CSV and JSON often require access to all relevant occurrences, rather than just a subset.\n", + "\n", + "In this tutorial, we present a technique that leverages SQL databases. By formulating a query based on the user's request, the model submits this query to the database, providing comprehensive results. This approach not only ensures users receive all pertinent information but also reduces the likelihood of exceeding token limits." + ] + }, + { + "cell_type": "markdown", + "id": "d513a9fe-1ac4-4610-b53c-6696e461f828", + "metadata": {}, + "source": [ + "## Prerequisites" + ] + }, + { + "cell_type": "markdown", + "id": "fbdb09aa-73d6-4cee-b3e7-5d6b243e3d58", + "metadata": {}, + "source": [ + "We assume you have access to Vertex AI and have enabled the necessary APIs.\n", + "\n", + "In this tutorial, we will be using Google Gemini Pro 1.5, which does not require deployment. However, if you prefer to use a different model, you can select one from the Model Garden via the console. This will allow you to add the model to your registry, create an endpoint (or utilize an existing one), and deploy the model—all in a single step. Here is a link for more information: [model deployment](https://cloud.google.com/vertex-ai/docs/general/deployment).\n", + "\n", + "Before we begin, you'll need to create a Vertex AI RAG Data Service Agent service account. To do this, go to the IAM section of the console. Ensure you check the box for \"Include Google-provided role grant.\" If the role is not listed, click \"Grant Access\" and add \"Vertex AI RAG Data Service Agent\" as a role." + ] + }, + { + "cell_type": "markdown", + "id": "eeb24716-0db8-462f-8faa-0b8ad9d61feb", + "metadata": { + "tags": [] + }, + "source": [ + "## Learning objectives" + ] + }, + { + "cell_type": "markdown", + "id": "f5216a43-31ee-4c3e-821a-4b81f4be8e54", + "metadata": {}, + "source": [ + "In this tutorial you will learn about:\n", + "- How to set up a BigQuery dataset and a table.\n", + "- How to load data to a BigQuery table.\n", + "- How to use the langchain ChatVertexAI agent to extract information froim the table. \n" + ] + }, + { + "cell_type": "markdown", + "id": "348910aa-9802-4318-8fcb-86603158923b", + "metadata": { + "tags": [] + }, + "source": [ + "## Pricing" + ] + }, + { + "cell_type": "markdown", + "id": "0dca31b2-880d-4cbc-bf76-ede96998e371", + "metadata": {}, + "source": [ + "
    \n", + "
  • BigQuery Standard 1TB Storage.
  • \n", + "
  • Model for Chat: 5 request per day. Average Input and Output characters 5,000.
  • \n", + "
  • Total $23.89 per month.
  • \n", + "
" + ] + }, + { + "cell_type": "markdown", + "id": "c7d9e3ee-1c7e-4baa-a818-c13330c665dd", + "metadata": {}, + "source": [ + "## Get Start" + ] + }, + { + "cell_type": "markdown", + "id": "137defcd-fd2b-421d-9b2e-fe2b878fb325", + "metadata": {}, + "source": [ + "### Install Packages" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "55c14a38-ba51-466d-90b5-1512f5a7fae0", + "metadata": { + "tags": [] + }, + "outputs": [], + "source": [ + "!pip install google-cloud-bigquery\n", + "!pip install gcloud\n", + "!pip install langchain\n", + "!pip install -U langchain-community\n", + "!pip install -qU langchain_google_vertexai\n", + "!pip install pandas-gbq" + ] + }, + { + "cell_type": "markdown", + "id": "64e22773-9a03-47de-9b6d-7946abcba722", + "metadata": {}, + "source": [ + "Set your project id, location, and bucket variables." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "7193a9ac-3169-45c1-9c5d-ba5e2f31458f", + "metadata": {}, + "outputs": [], + "source": [ + "project_id=''\n", + "location=' (e.g.us-east4)'\n", + "bucket = ''" + ] + }, + { + "cell_type": "markdown", + "id": "df0d9eda-ee54-4c4c-86e6-9166b2b82e2d", + "metadata": {}, + "source": [ + "Create a bucket where the data used for tutorial is going to be placed." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "7b583234-2d92-4e5b-841d-c7d79dd87d45", + "metadata": {}, + "outputs": [], + "source": [ + "# make bucket\n", + "!gsutil mb -l {location} gs://{bucket}" + ] + }, + { + "cell_type": "markdown", + "id": "5781a07c-84ed-416c-a782-b1dc881e5c18", + "metadata": {}, + "source": [ + "Provide the names of the bucket name, source file name path, and destination blob name to upload the CSV source file to the bucket." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "4829094e-61a6-4dd7-854f-6b21dc19b1b7", + "metadata": {}, + "outputs": [], + "source": [ + "from google.cloud import storage\n", + "\n", + "# The ID of your GCS bucket\n", + "bucket_name = bucket\n", + " # The path to your file to upload\n", + "source_file_name = \"\"\n", + " # The ID of your GCS object\n", + "destination_blob_name = \"\"\n", + "\n", + "storage_client = storage.Client()\n", + "bucket = storage_client.bucket(bucket_name)\n", + "blob = bucket.blob(destination_blob_name)\n", + "blob.upload_from_filename(source_file_name)\n", + "\n", + "print(\n", + " \"File {} uploaded to {}.\".format(\n", + " source_file_name, destination_blob_name\n", + " )\n", + ")\n" + ] + }, + { + "cell_type": "markdown", + "id": "7a00dc83-8713-4ea5-9345-08a9f3d785d0", + "metadata": {}, + "source": [ + "The next step is to create a database in BigQuery. To accomplish this, we need to define a dataset_id and construct a dataset object that will be sent to the API for creation. Note that in BigQuery, a dataset is analogous to a database." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "1750e13d-ec05-4335-82a2-ffe54c5b5f32", + "metadata": {}, + "outputs": [], + "source": [ + "from google.cloud import bigquery\n", + "\n", + "client = bigquery.Client()\n", + "\n", + "# Set dataset_id to the ID of the dataset to create.\n", + "dataset_id = project_id.''\n", + "\n", + "# Construct a full Dataset object to send to the API.\n", + "dataset = bigquery.Dataset(dataset_id)\n", + "\n", + "# Specify the geographic location where the dataset should reside.\n", + "dataset.location = location\n", + "\n", + "# Send the dataset to the API for creation, with an explicit timeout.\n", + "# Raises google.api_core.exceptions.Conflict if the Dataset already\n", + "# exists within the project.\n", + "dataset = client.create_dataset(dataset, timeout=30) # Make an API request.\n", + "print(\"Created dataset {}.{}\".format(client.project, dataset.dataset_id))" + ] + }, + { + "cell_type": "markdown", + "id": "544c191c-2bc0-4a47-a0a7-cea81035597a", + "metadata": {}, + "source": [ + "Once the dataset is created, the next step is to create a table. First, we define a table ID, followed by outlining the table's schema. Finally, we send an API request to create the table within the dataset established in the previous step." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "68206b14-1b5e-49c3-9f07-52ba6c70a93e", + "metadata": {}, + "outputs": [], + "source": [ + "# Set table_id to the ID of the table to create.\n", + "\n", + "client = bigquery.Client()\n", + "\n", + "table_id = project_id.dataset_id.\"
\"\n", + "\n", + "schema = [\n", + " bigquery.SchemaField(\"id\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"age\", \"INTEGER\", mode=\"NULLABLE\"), \n", + " bigquery.SchemaField(\"gender\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"height\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"weight\", \"FLOAT\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"ap_hi\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"ap_lo\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"cholesterol\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"gluc\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"smoke\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"alco\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"active\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"cardio\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"ageinyr\", \"INTEGER\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"bmi\", \"FLOAT\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"bmicat\", \"STRING\", mode=\"NULLABLE\"),\n", + " bigquery.SchemaField(\"agegroup\", \"STRING\", mode=\"NULLABLE\"),\n", + "\n", + "]\n", + "\n", + "table = bigquery.Table(table_id, schema=schema)\n", + "table = client.create_table(table) # Make an API request.\n", + "print(\n", + " \"Created table {}.{}.{}\".format(table.project, table.dataset_id, table.table_id)\n", + ")" + ] + }, + { + "cell_type": "markdown", + "id": "6e97b4c6-429c-4d36-a4c1-3963b079af6a", + "metadata": {}, + "source": [ + "This step is optional. We can make an API request to verify whether the dataset has been successfully created under our project." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "72f11cfd-8b20-48bf-b157-aaf791cd3a9f", + "metadata": {}, + "outputs": [], + "source": [ + "datasets = list(client.list_datasets()) # Make an API request.\n", + "project = project_id\n", + "\n", + "if datasets:\n", + " print(\"Datasets in project {}:\".format(project))\n", + " for dataset in datasets:\n", + " print(\"\\t{}\".format(dataset.dataset_id))\n", + "else:\n", + " print(\"{} project does not contain any datasets.\".format(project))" + ] + }, + { + "cell_type": "markdown", + "id": "dad4f96d-ae52-43f2-8290-c04b25960ea8", + "metadata": {}, + "source": [ + "We can perform a similar check to confirm that the table we created is located within the appropriate dataset." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "80ea826c-9d70-4268-b4db-a657d5cdcaf0", + "metadata": {}, + "outputs": [], + "source": [ + "# Set dataset_id to the ID of the dataset to fetch.\n", + "\n", + "dataset = client.get_dataset(dataset_id) # Make an API request.\n", + "\n", + "full_dataset_id = \"{}.{}\".format(dataset.project, dataset.dataset_id)\n", + "friendly_name = dataset.friendly_name\n", + "print(\n", + " \"Got dataset '{}' with friendly_name '{}'.\".format(\n", + " full_dataset_id, friendly_name\n", + " )\n", + ")\n", + "\n", + "# View dataset properties.\n", + "print(\"Description: {}\".format(dataset.description))\n", + "print(\"Labels:\")\n", + "labels = dataset.labels\n", + "if labels:\n", + " for label, value in labels.items():\n", + " print(\"\\t{}: {}\".format(label, value))\n", + "else:\n", + " print(\"\\tDataset has no labels defined.\")\n", + "\n", + "# View tables in dataset.\n", + "print(\"Tables:\")\n", + "tables = list(client.list_tables(dataset)) # Make an API request(s).\n", + "if tables:\n", + " for table in tables:\n", + " print(\"\\t{}\".format(table.table_id))\n", + "else:\n", + " print(\"\\tThis dataset does not contain any tables.\")" + ] + }, + { + "cell_type": "markdown", + "id": "736080b9-0cd9-44fd-9580-dadb10a22a59", + "metadata": { + "tags": [] + }, + "source": [ + "To upload the data from our file stored in the bucket, we first need to create a DataFrame. In this example, we will be using a file that contains health screening data." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "b614cc1a-b318-4364-8d47-1dea9700c792", + "metadata": {}, + "outputs": [], + "source": [ + "from google.cloud import storage\n", + "import pandas as pd\n", + "from io import StringIO\n", + "\n", + "# Initialize a client\n", + "client = storage.Client()\n", + "my_bucket = bucket\n", + "storage_client = storage.Client()\n", + "bucket = storage_client.get_bucket(my_bucket)\n", + "blob = bucket.blob(destination_blob_name)\n", + "path = \"\"\n", + "df = pd.read_csv(path)\n", + "df.head()" + ] + }, + { + "cell_type": "markdown", + "id": "ad42d610-daa6-4d92-ac14-2d90349dafe1", + "metadata": {}, + "source": [ + "We need to use the \"to_gbq\" function from the pandas-gbq library, which allows us to write a Pandas DataFrame to a Google BigQuery table. This enables us to populate our BigQuery table with the data from the DataFrame." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "7cfd1d66-a0fb-403d-a9b1-27250cc81f91", + "metadata": {}, + "outputs": [], + "source": [ + "df.to_gbq(table_id, project_id=project_id)\n", + "\n", + "from google.cloud import storage\n", + "import pandas as pd\n", + "from io import StringIO\n", + "\n", + "client.load_table_from_dataframe(df, table_id).result()" + ] + }, + { + "cell_type": "markdown", + "id": "22b4dd65-9bd9-48ef-9c98-85e3e9a82701", + "metadata": {}, + "source": [ + "This step is also optional. We can execute a simple query on the BigQuery table to count the number of records." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "29e91fc1-735f-4d6c-9fcf-3a2d745363c1", + "metadata": {}, + "outputs": [], + "source": [ + "query = \"\"\"\n", + " SELECT count(gender) as cgender\n", + " FROM `project_id.dataset_id.table_id`\n", + "\"\"\"\n", + "\n", + "# Execute the query\n", + "query_job = client.query(query)\n", + "\n", + "# Fetch the results\n", + "results = query_job.result()\n", + "\n", + "# Iterate through the rows\n", + "for row in results:\n", + " #print(f\"gender: {row.name}, age: {row.age}\")\n", + " print(f\"cgender: {row.cgender}\")\n", + "\n", + "try:\n", + " query_job = client.query(query)\n", + " results = query_job.result()\n", + " print(results)\n", + "except Exception as e:\n", + " print(f\"Error executing query: {str(e)}\")" + ] + }, + { + "cell_type": "markdown", + "id": "ce65ead3-694d-4d46-9376-38d35b8d0852", + "metadata": {}, + "source": [ + "cgender: 139920\n", + "" + ] + }, + { + "cell_type": "markdown", + "id": "b6fb53b8-9f93-4f37-8730-33c5871720fc", + "metadata": {}, + "source": [ + "To interact with the BigQuery table using a Pythonic domain language, we utilize SQLAlchemy. SQLAlchemy is a Python SQL toolkit that enables developers to access and manage SQL databases, allowing users to write queries as strings or chain Python objects for similar queries. However, to do this, we need specific credentials, which can be accessed through a service account file. For more information on how to create a service account file, please visit the following link: [Service Account Creation](https://cloud.google.com/iam/docs/service-accounts-create#python)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "d1d29b90-7bcd-46b8-82b9-dc5e58f9edee", + "metadata": {}, + "outputs": [], + "source": [ + "from google.cloud import bigquery\n", + "from sqlalchemy import *\n", + "from sqlalchemy.engine import create_engine\n", + "from sqlalchemy.schema import *\n", + "import os\n", + "from langchain_community.agent_toolkits import create_sql_agent\n", + "from langchain.agents import create_sql_agent\n", + "from langchain.agents.agent_toolkits import SQLDatabaseToolkit\n", + "from langchain.sql_database import SQLDatabase\n", + "service_account_file = \"\" # Change to where your service account key file is located\n", + "project_id = project_id\n", + "dataset = \"\"\n", + "table = \"
\"\n", + "sqlalchemy_url = f'bigquery://{project_id}/{dataset}?credentials_path={service_account_file}'\n", + "print(sqlalchemy_url)" + ] + }, + { + "cell_type": "markdown", + "id": "9194cc1d-7ef1-4c4e-8e65-e7d9ada5be24", + "metadata": {}, + "source": [ + "Next, we import the __ChatVertexAI__ agent from Langchain and configure it with the appropriate hyperparameters. In this instance, we are using the __gemini-1.5-pro__ LLM model, after which we create the SQL agent to enable querying the BigQuery table using a natural language string as a prompt." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "ee1899a2-dd79-4cf4-a02d-c7c35b99040b", + "metadata": {}, + "outputs": [], + "source": [ + "from langchain_google_vertexai import VertexAI, ChatVertexAI\n", + "\n", + "llm = ChatVertexAI(\n", + " model=\"gemini-1.5-pro\",\n", + " temperature=0,\n", + " max_tokens=8190,\n", + " timeout=None,\n", + " max_retries=2,\n", + ")\n", + "db = SQLDatabase.from_uri(sqlalchemy_url)\n", + "toolkit = SQLDatabaseToolkit(db=db, llm=llm)\n", + "agent_executor = create_sql_agent(\n", + " llm=llm,\n", + " toolkit=toolkit,\n", + " verbose=True,\n", + " top_k=100000\n", + " )\n" + ] + }, + { + "cell_type": "markdown", + "id": "f3551426-8644-48ba-b8c1-bf7d5eb843b3", + "metadata": {}, + "source": [ + "For our initial query, we check the number of rows to compare the result with what we obtained in a previous step using a simple SQL query against the BigQuery table. We can confirm that we received the same number of rows: 139,920." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "e9753533-e82b-47af-936a-a401d512308a", + "metadata": {}, + "outputs": [], + "source": [ + "agent_executor.run(\"How many rows are in the table screening? \")" + ] + }, + { + "cell_type": "markdown", + "id": "099154a1-34e2-4f6b-b146-cec6f5943cf7", + "metadata": {}, + "source": [ + "> Finished chain.\n", + "'The table screening has 139920 rows.'" + ] + }, + { + "cell_type": "markdown", + "id": "65f645ed-8efa-436f-b993-72c45a7aadbe", + "metadata": { + "tags": [] + }, + "source": [ + "We know the output of the model are correct. The image below is a reminder of what the actual results are calculated from." + ] + }, + { + "cell_type": "markdown", + "id": "8160ee6e-f95f-44c3-ab27-4415b44ba461", + "metadata": { + "tags": [] + }, + "source": [ + "![image.png](../../images/gcp_rag_structure_data_01.png)" + ] + }, + { + "cell_type": "markdown", + "id": "f2b62012-b2d2-48c1-ae69-6f8816065551", + "metadata": { + "tags": [] + }, + "source": [ + "Next, we pose a question that requires applying a filter, and we successfully obtain the accurate number of obese individuals in the table." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "5563d317-81ab-4a37-8225-e1b0dce20b36", + "metadata": {}, + "outputs": [], + "source": [ + "agent_executor.run(\"How many obese are in the table screen?\")" + ] + }, + { + "cell_type": "markdown", + "id": "40e1846d-1114-4ba1-b35a-65b47a838468", + "metadata": {}, + "source": [ + "> Finished chain.\n", + "'There are 37204 obese people in the table.'" + ] + }, + { + "cell_type": "markdown", + "id": "b96ea414-3f2d-4e66-ab0b-a034dfe507b0", + "metadata": {}, + "source": [ + "As a more complex query, we inquired about the number of female smokers, and the SQL agent accurately returned the answer of 1,626 female smokers." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "a456ff90-a3a5-4ee4-81da-fba4fcdf7f2d", + "metadata": {}, + "outputs": [], + "source": [ + "agent_executor.run(\"How many smokers are female in the table screen? the value 1 in smoke means that is a smoker and the value 2 in gender means that is a female\")" + ] + }, + { + "cell_type": "markdown", + "id": "7ea7810f-9701-4cb0-9430-181a57929884", + "metadata": {}, + "source": [ + "> Finished chain.\n", + "'There are 1626 female smokers in the table.'" + ] + }, + { + "cell_type": "markdown", + "id": "ca0fb657-19e0-4488-b9f6-25ca3971064f", + "metadata": { + "tags": [] + }, + "source": [ + "## Conclusion" + ] + }, + { + "cell_type": "markdown", + "id": "7f85e560-231f-4841-b1d0-334326e4b347", + "metadata": {}, + "source": [ + "You have learned how to create a dataset and a table in BigQuery, as well as how to set up an agent that enables you to query the table using natural language instead of SQL queries, allowing you to obtain the results you need." + ] + }, + { + "cell_type": "markdown", + "id": "41573388-7feb-44b6-9e8a-0511cc8ae62e", + "metadata": {}, + "source": [ + "## Clean Up" + ] + }, + { + "cell_type": "markdown", + "id": "1f0edc0b-f2cc-478c-855f-9acff3792f4d", + "metadata": {}, + "source": [ + "Please remember to delete or stop your Jupyter notebook and delete your data store to prevent incurring charges. And if you have created any other services like buckets, please remember to delete them as well." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "fe73dde2-056a-46a3-bacb-0a6671f7128f", + "metadata": {}, + "outputs": [], + "source": [] + } + ], + "metadata": { + "environment": { + "kernel": "python3", + "name": "common-cpu.m125", + "type": "gcloud", + "uri": "us-docker.pkg.dev/deeplearning-platform-release/gcr.io/base-cpu:m125" + }, + "kernelspec": { + "display_name": "Python 3 (Local)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.10.15" + } + }, + "nbformat": 4, + "nbformat_minor": 5 +}