LoginSignup
5
4

Oracle Database 23ai FreeとAPEXでRAGを使った生成AIアプリをローコード開発してみた (アプリ実装 後編)

Last updated at Posted at 2024-05-07

前回はアプリ実装 前編として、非構造化データをアップロードし、Embedding結果を保存する機能を実装しました。
Oracle Database 23ai FreeとAPEXでRAGを使った生成AIアプリをローコード開発してみた (アプリ実装 前編)
今回は後編にあたる生成AIに質問し、RAGを使った回答を表示させる機能を実装します。

  • 前編:任意のファイル (PDFやOfficeファイル) をアップロードしてEmbeddingするページ
  • 後編:質問文でベクトル検索し、質問文と検索結果をもとにした生成AIの回答を表示するページ

なおアプリのexportファイルを以下に置いています。
https://github.com/mago1chi/apex/blob/main/ragapp_cohere.sql
今回のアプリ実装は作業数が多いため、すぐに動くものを見たい方は上記ファイルをご自分のAPEX環境にimportしてお試しください。
その場合でも、以下記事の作業は事前に行う必要がありますのでご注意ください。

後編で作成するページは以下のようなイメージになります。
WS000089.JPG

こちらは以下の処理を行います。

  • 質問フォームに入力した質問文をEmbeddingし、登録済みファイルにベクトル検索を実行
  • ベクトル検索の際は以下の付加情報も利用
    • 取得行数: 類似するコンテンツのうち上位何件を利用するか
    • 保証ドキュメント数: マルチベクトル検索を使うにあたり、上位何件のドキュメントを含めるか
    • カテゴリ: ベクトル検索に含めるドキュメントをどのカテゴリのものに絞るか
    • 開始日付、終了日付: ベクトル検索に含めるドキュメントをいつ作成されたものに絞るか
  • マークダウン対応した回答欄に、生成AIからの回答を表示
  • 引用元欄にベクトル検索の結果RAGに使われたテキスト、対応ファイル名を表示

前編と同様、以降は手順を記載しつつ、実装に使うベクトル検索機能 (AI Vector Search) の説明も簡単に挟みたいと思います。
ちなみにRAG実装にあたり、LangChainなどのオーケストレーション・フレームワークは使用しておりません。
本例ではベクトル検索の結果をコンテキストとしてプロンプトに埋め込み、質問文と一緒にLLMへ連携する方式を取っています。

APEXにウォレットを設定

外部サービスのREST APIを使うには、事前にルート認証局を設定したウォレットを作成する必要があります。
まずCLIでウォレット作成を行い、その後APEXにウォレットを読み込ませるための設定を行います。
手順は下記サイトを参考にしました。
https://apexugj.blogspot.com/2023/04/create-wallet-for-apex.html

## rootユーザで実行
## 必要なパッケージを導入
$ sudo yum install ca-certificates

## 以降はoracleユーザで実行
$ sudo su - oracle
$ export ORACLE_WALLET_PATH=/opt/oracle/ca_wallet
$ export ORACLE_WALLET_PWD=WalletPasswd123
$ mkdir -p ${ORACLE_WALLET_PATH}

## ウォレット作成
$ orapki wallet create -wallet ${ORACLE_WALLET_PATH} -pwd ${ORACLE_WALLET_PWD} -auto_login

## 証明書を取得
$ cd ${ORACLE_WALLET_PATH}
$ wget https://curl.haxx.se/ca/cacert.pem

## 認証局ごとに証明書を分割しファイルに保存
$ awk 'split_after == 1 {n++;split_after=0} /-----END CERTIFICATE-----/ {split_after=1} {print > "cert" n ".pem"}' < cacert.pem

## 分割した各認証局の証明書をウォレットにロード
$ for i in /opt/oracle/ca_wallet/cert*.pem
do
  orapki wallet add -wallet ${ORACLE_WALLET_PATH} -trusted_cert -cert "$i" -pwd ${ORACLE_WALLET_PWD}
done

次に作成したウォレットを読み込めるよう、APEXに設定を施します。
ワークスペースへログインしている場合は、一度サインアウトします。
WS000051.JPG

ログイン画面の下段にある「管理」からAPEXインスタンス管理のログイン画面へ遷移します。
管理ユーザでログインします。
WS000052.JPG

「インスタンスの管理」を選択し、「インスタンスの設定」へ進みます。
WS000053.JPG
WS000054.JPG

「ウォレット」タブを選択し、赤枠を画像の通りに編集します。
WS000055.JPG

「電子メール」タブを選択します。
こちらはウォレット設定とは関係ないですが、「電子メール」タブの内容も設定しないと「変更の適用」がエラーになるため対応が必要です。
「デフォルトURL」を選択し、空欄となっている2箇所を埋めます。
WS000056.JPG

最後に「変更の適用」を選択します。
以上でウォレット設定が完了しました。

Web資格証明の設定

CohereのLLMをREST API経由で利用するには、事前に資格証明 (利用に必要なAPIキー) をAPEX上で設定する必要があります。
まずワークスペースに再度ログインします。
メニューの「アプリケーション・ビルダー」にある「ワークスペース・ユーティリティ」の「すべてのワークスペース・ユーティリティ」を選択します。
WS000077.JPG

「Web資格証明」を選択します。
WS000078.JPG

「作成」を選択します。
WS000079.JPG

赤枠を画像の通りに編集します。
WS000080.JPG

入力する値は以下の通りです。

名前: Credential for Cohere
静的ID: CREDENTIAL_FOR_COHERE
認証タイプ: HTTPヘッダー
資格証明名: Authorization
資格証明シークレット: bearer your_api_key
URLに対して有効: https://api.cohere.ai/

「資格証明シークレット」には「bearer」のあとに半角スペースを1つ入れ、ご自分のAPIキーを入力してください。
また静的IDは後述するPL/SQLコードの中で利用します。
最後に「作成」ボタンを押下します。

ユーザ・インターフェイスの編集

最初に空白ページとして作成した「RAG」という名前のページについて、ユーザ・インターフェイスを編集します。
アプリケーション・ビルダーのトップ画面から「4 - RAG」を選択します。
WS000057.JPG

ページ・デザイナを開いたら下段の「アイテム」タブにある「テキスト・フィールド」を、「BODY」エリアにD&Dで追加します。
WS000058.JPG

右ペイン赤枠を画像の通りに編集します。
WS000059.JPG

右ペインを下にスクロールし、赤枠を画像の通りに編集します。
WS000060.JPG

これで質問フォームを入力必須に設定します。

以降は同じようにアイテムやボタンをD&DでBODYエリアに追加していきます。
次は「数値フィールド」をBODYエリア「P4_QUESTION」の下に追加します。
右ペイン赤枠を画像の通りに編集します。
WS000061.JPG

右ペインを下にスクロールし、赤枠を画像の通りに編集します。
WS000062.JPG

さらに右ペインを下にスクロールし、赤枠を画像の通りに編集します。
WS000062-2.JPG

本フォームの値は「類似するコンテンツのうち上位何件を利用するか」を制御します。
入力できる範囲を 1~5 に制限し、デフォルト値を 3 にしています。
また入力を必須に設定しています。

「数値フィールド」をBODYエリア「P4_NUM」の右に追加します。
右ペイン赤枠を画像の通りに編集します。
WS000063.JPG

本フォームの値は「マルチベクトル検索を使うにあたり、上位何件のドキュメントを含めるか」を制御します。
マルチベクトル検索については後述します。

「ポップアップLOV」をBODYエリア「P4_DOCNUM」の右に追加します。
右ペイン赤枠を画像の通りに編集します。
WS000064.JPG

右ペインを下にスクロールし、赤枠を画像の通りに編集します。
WS000065.JPG

さらに右ペインを下にスクロールし、赤枠を画像の通りに編集します。
WS000066.JPG

「SQL問合せ」に入力するSQLは以下の通りです。

SELECT category d, category v FROM documents GROUP BY category order by 1;

本フォームの値は「ベクトル検索に含めるドキュメントをどのカテゴリに絞るか」を制御します。
ポップアップLOVについては前編でも利用した、選択リストの一種になります。

「日付ピッカー」をBODYエリア「P4_CATEGORY」の右に追加します。
右ペイン赤枠を画像の通りに編集します。
WS000067.JPG

右ペインを下にスクロールし、赤枠を画像の通りに編集します。
WS000068.JPG

さらに右ペインを下にスクロールし、赤枠を画像の通りに編集します。
WS000069.JPG

「SQL問合せ」に入力するSQLは以下の通りです。

SELECT sysdate - 365 FROM dual;

本フォームの値は「ベクトル検索に含めるドキュメントをいつ作成されたものに絞るか」を制御します。
デフォルト値はその日の日付から1年前を設定しています。

再度「日付ピッカー」をBODYエリア「P4_BEGIN_DATE」の右に追加します。
右ペイン赤枠を画像の通りに編集します。
WS000070.JPG

右ペインを下にスクロールし、赤枠を画像の通りに編集します。
WS000071.JPG

さらに右ペインを下にスクロールし、赤枠を画像の通りに編集します。
WS000072.JPG

「SQL問合せ」に入力するSQLは以下の通りです。

SELECT sysdate FROM dual;

ここで「P4_BEGIN_DATE」の値が「P4_END_DATE」より大きくならないよう、検証する処理を追加します。
左ペインの「プロセス」タブを選択します。
「検証中」の上で右クリックし「検証の作成」を選択します。
WS000072-2.JPG

右ペイン赤枠を画像の通りに編集します。
WS000072-3.JPG

「PL/SQL式」に入力するコードは以下の通りです。

:P4_BEGIN_DATE <= :P4_END_DATE

下段の「ボタン」タブを選択し、「icon」と書かれたボタンをBODYエリア「P4_END_DATE」の右に追加します。
右ペイン赤枠を画像の通りに編集します。
WS000073.JPG

ボタンの「テンプレート・オプション」についても、赤枠を画像の通りに編集します。
WS000073-2.JPG

ボタンの配置をCSSで調整します。
左ペインの「ページ4: RAG」を選択し、右ペイン赤枠を画像の通りに編集します。
WS000074.JPG

「インライン」に入力するCSSは以下の通りです。

.myc-aligncenter {
  display: flex;
  align-items: center;
}

左ペインの「P4_NUM」を選択し、右ペイン赤枠を画像の通りに編集します。
WS000075.JPG

「行CSSクラス」に入力する値は以下の通りです。

myc-aligncenter

生成AIからの回答を表示するアイテムを追加します。
「リッチ・テキスト・エディタ」をBODYエリアの新しい行へ追加します。
右ペイン赤枠を画像の通りに編集します。
WS000084.JPG

右ペインを下にスクロールし、赤枠を画像の通りに編集します。
WS000085.JPG

このアイテムはユーザ側で編集しない想定のため、読取り専用にしています。

同じように引用元テキスト表示用のアイテムを追加します。
「リッチ・テキスト・エディタ」をBODYエリアの新しい行へ追加します。
右ペイン赤枠を画像の通りに編集します。
WS000086.JPG

右ペインを下にスクロールし、赤枠を画像の通りに編集します。
WS000087.JPG

このアイテムには引用元のテキスト、および紐づくファイル名を表示します。
その際、HTML形式で整形したテーブルを読み込ませますので、「書式」を「HTML」にしています。
またこのアイテムもユーザ側で編集しない想定のため、読取り専用にしています。

PL/SQLを使ったベクトル検索、生成AI連携の実装

最後に生成AIと連携する内部処理をPL/SQLで実装します。
左ペインの「プロセス」タブを選択し、新しいプロセスを作成します。
右ペイン赤枠を画像の通りに編集します。
WS000082.JPG

「PL/SQLコード」に入力するPL/SQLは以下の通りです。

DECLARE
    cohere_params clob;
    l_cohere_rest_url VARCHAR2(4000) := 'https://api.cohere.ai/v1/chat'; 
    -- Web資格証明に設定した「静的ID」を入力
    l_web_cred CONSTANT VARCHAR2(50) := 'CREDENTIAL_FOR_COHERE';
    l_context CLOB;
    l_prompt CLOB;
    l_response_json CLOB;
    l_body CLOB;
    l_citation CLOB;
    CURSOR C1  IS 
    SELECT jt.* 
    FROM   JSON_TABLE(l_response_json, '$' 
        COLUMNS (text VARCHAR2(4000)  PATH '$.text' )) jt; 
BEGIN
    -- Cohereの embed-multilingual-v3.0 を使ってEmbeddingするためのパラメータ
    cohere_params := '
    { 
      "provider": "cohere",
      "credential_name": "COHERE_CRED", 
      "url": "https://api.cohere.ai/v1/embed",
      "model": "embed-multilingual-v3.0",
      "input_type": "search_query"
    }';
    
    -- 引用元の情報を格納する変数を用意
    l_citation := '<table><tr><th>ファイル名</th><th>該当テキスト</th></tr>';
    
    -- ベクトル検索を実行し、RAGに使用するコンテキストを取得
    IF :P4_CATEGORY IS NULL THEN
        IF :P4_DOCNUM IS NULL THEN
            -- カテゴリ、保証ドキュメント数の指定がない場合のクエリ
            FOR rec IN (SELECT doc.doc_filename doc_filename, em.embed_data embed_data
                FROM documents doc, embed em 
                WHERE doc.id = em.doc_id AND TRUNC(doc.doc_lastupd) BETWEEN :P4_BEGIN_DATE AND :P4_END_DATE
                ORDER BY VECTOR_DISTANCE(em.embedding, DBMS_VECTOR.UTL_TO_EMBEDDING(:P4_QUESTION, JSON(cohere_params)), COSINE)
                FETCH EXACT FIRST :P4_NUM ROWS ONLY)
            LOOP
                -- プロンプトに埋め込むコンテキストを追記
                l_context := l_context || rec.embed_data;
                -- 引用元情報を追記
                l_citation := l_citation || '<tr><td>' || rec.doc_filename || '</td><td>' || rec.embed_data || '</td></tr>';
            END LOOP;
        ELSE
            -- カテゴリの指定がなく、保証ドキュメント数の指定がある場合のクエリ
            FOR rec IN (SELECT doc.doc_filename doc_filename, em.embed_data embed_data
                FROM documents doc, embed em 
                WHERE doc.id = em.doc_id AND TRUNC(doc.doc_lastupd) BETWEEN :P4_BEGIN_DATE AND :P4_END_DATE
                ORDER BY VECTOR_DISTANCE(em.embedding, DBMS_VECTOR.UTL_TO_EMBEDDING(:P4_QUESTION, JSON(cohere_params)), COSINE)
                FETCH EXACT FIRST :P4_DOCNUM PARTITIONS BY doc.id, :P4_NUM ROWS ONLY)
            LOOP
                -- プロンプトに埋め込むコンテキストを追記
                l_context := l_context || rec.embed_data;
                -- 引用元情報を追記
                l_citation := l_citation || '<tr><td>' || rec.doc_filename || '</td><td>' || rec.embed_data || '</td></tr>';
            END LOOP;
        END IF;
    ELSE
        -- カテゴリの指定があり、保証ドキュメント数の指定がない場合のクエリ
        IF :P4_DOCNUM IS NULL THEN
            FOR rec IN (SELECT doc.doc_filename doc_filename, em.embed_data embed_data
                FROM documents doc, embed em 
                WHERE doc.id = em.doc_id AND 
                doc.category IN (SELECT column_value FROM APEX_STRING.SPLIT(:P4_CATEGORY,':')) AND 
                TRUNC(doc.doc_lastupd) BETWEEN :P4_BEGIN_DATE AND :P4_END_DATE
                ORDER BY VECTOR_DISTANCE(em.embedding, DBMS_VECTOR.UTL_TO_EMBEDDING(:P4_QUESTION, JSON(cohere_params)), COSINE)
                FETCH EXACT FIRST :P4_NUM ROWS ONLY)
            LOOP
                -- プロンプトに埋め込むコンテキストを追記
                l_context := l_context || rec.embed_data;
                -- 引用元情報を追記
                l_citation := l_citation || '<tr><td>' || rec.doc_filename || '</td><td>' || rec.embed_data || '</td></tr>';
            END LOOP;
        ELSE
            -- カテゴリ、保証ドキュメント数の指定がある場合のクエリ
            FOR rec IN (SELECT doc.doc_filename doc_filename, em.embed_data embed_data
                FROM documents doc, embed em 
                WHERE doc.id = em.doc_id AND 
                doc.category IN (SELECT column_value FROM APEX_STRING.SPLIT(:P4_CATEGORY,':')) AND 
                TRUNC(doc.doc_lastupd) BETWEEN :P4_BEGIN_DATE AND :P4_END_DATE
                ORDER BY VECTOR_DISTANCE(em.embedding, DBMS_VECTOR.UTL_TO_EMBEDDING(:P4_QUESTION, JSON(cohere_params)), COSINE)
                FETCH EXACT FIRST :P4_DOCNUM PARTITIONS BY doc.id, :P4_NUM ROWS ONLY)
            LOOP
                -- プロンプトに埋め込むコンテキストを追記
                l_context := l_context || rec.embed_data;
                -- 引用元情報を追記
                l_citation := l_citation || '<tr><td>' || rec.doc_filename || '</td><td>' || rec.embed_data || '</td></tr>';
            END LOOP;
        END IF;
    END IF;
    
    -- プロンプト生成
    -- ベクトル検索結果をコンテキストとして埋め込むことでRAGを実装
    l_prompt := 'Answer the following question using the supplied context assuming you are a subject matter expert. Question: ' || :P4_QUESTION || ' Context: ' || l_context; 
    
    -- LLMにPOSTリクエストするためのBody設定
    l_body := '{ "message": "'|| APEX_ESCAPE.JSON(l_prompt) ||'" }';

    -- LLMにPOSTリクエストするためのHeader設定
    APEX_WEB_SERVICE.G_REQUEST_HEADERS.DELETE; 
    APEX_WEB_SERVICE.G_REQUEST_HEADERS(1).NAME  := 'Content-Type'; 
    APEX_WEB_SERVICE.G_REQUEST_HEADERS(1).VALUE := 'application/json';  

    -- LLMにREST API発行
    l_response_json := APEX_WEB_SERVICE.MAKE_REST_REQUEST 
    (p_url                  => l_cohere_rest_url, 
        p_http_method          => 'POST', 
        p_body                 => l_body, 
        p_credential_static_id => l_web_cred); 

    -- LLMからの応答を回答表示用アイテムに格納
    FOR row_1 IN C1 LOOP
        :P4_ANSWER := row_1.text; 
    END LOOP;
    
    -- 引用元情報を引用元表示用アイテムに格納
    :P4_CITATION := l_citation || '</table>';
END;

コードの大まかな内容はコメントをご確認ください。
AI Vector Searchの機能を使っている箇所は別途解説したいと思います。
まず 35~36行目 の構文は AI Vector Search 用に23aiで新たに追加された機能を使っています。

ORDER BY VECTOR_DISTANCE(em.embedding, DBMS_VECTOR.UTL_TO_EMBEDDING(:P4_QUESTION, JSON(cohere_params)), COSINE)
FETCH EXACT FIRST :P4_NUM ROWS ONLY)

ORDER BY VECTOR_DISTANCE は第一引数と第二引数のベクトルデータ間の類似度を算出するものです。
第二引数のプロシージャは前編でも出てきました。
第三引数は類似度の算出方法を指定しますが、本例では「COSINE」(コサイン類似度) を使っています。
なお選択できるパラメータ (類似度の算出方法) は以下の通りです。

  • COSINE: コサイン類似度 (デフォルト値)
  • DOT: ドット積
  • EUCLIDEAN: ユークリッド距離
  • EUCLIDEAN_SQUARED: 平方ユークリッド距離 (ユークリッド距離の平方根なし版)
  • HAMMING: ハミング距離
  • MANHATTAN: マンハッタン距離

FETCH EXACT FIRST :P4_NUM ROWS ONLY は類似度の高い順に上から「:P4_NUM」件の結果を返す、というものです。
また最後に参考として記載しますが、こちらはベクトル索引を使わない場合の構文となります。

より詳細は下記マニュアルをご参照ください。

48行目 は上記と少し異なる構文ですが、こちらは「マルチベクトル検索」を使う場合の構文です。

FETCH EXACT FIRST :P4_DOCNUM PARTITIONS BY doc.id, :P4_NUM ROWS ONLY)

こちらの解釈は「類似度の高い上位 :P4_DOCNUM 件のドキュメントそれぞれから、類似度の高い上位 :P4_NUM 件のテキストを抽出する」という意味になります。
そのため出力される結果の行数は :P4_DOCNUM x :P4_NUM 件になります。
より詳細は下記マニュアルをご参照ください。

最後に右ペインを下にスクロールし、赤枠を画像の通りに編集します。
WS000083.JPG

これで「検索」ボタンを押下すると、上述したPL/SQLコードが実行されます。

以上でアプリ実装は完了です。
今回作成した機能の使用感は以下のようなイメージです。
WS000090.JPG

本例では以下のフィルタリングを施した上でクエリを投げています。

  • 取得行数: 上位3件
  • 保証ドキュメント数: 上位2件
  • カテゴリ: Linux
  • ファイル作成日: 2023/5/5~2024/5/4

マルチベクトル検索を使っているため、カテゴリ「Linux」に含まれるドキュメントのうち上位2件からそれぞれ、3件ずつ関連度の高いテキストが取得されています。
またそれらテキストをコンテキストとしてプロンプトに埋め込み、生成AIに質問を投げています。
結果として生成AIの回答は、コンテキストに含まれる文章を取り入れた回答になっています。

(参考) EmbeddingにONNX形式でimportしたモデルを使う場合

ご参考までに、ONNX形式でimportしたモデルを使ってベクトル検索し、検索結果をLLMに連携するコードを掲載します。
importしたモデルを使うことで、DB内でEmbedding処理を実行できます。
LLMも閉域網内のサーバに自前で用意すれば、インターネットを経由しない形でRAGシステムを構築できます。

DECLARE
    l_cohere_rest_url VARCHAR2(4000) := 'https://api.cohere.ai/v1/chat'; 
    -- Web資格証明に設定した「静的ID」を入力
    l_web_cred CONSTANT VARCHAR2(50) := 'CREDENTIAL_FOR_COHERE';
    l_context CLOB;
    l_prompt CLOB;
    l_response_json CLOB;
    l_body CLOB;
    l_citation CLOB;
    CURSOR C1  IS 
    SELECT jt.* 
    FROM   JSON_TABLE(l_response_json, '$' 
        COLUMNS (text VARCHAR2(4000)  PATH '$.text' )) jt; 
BEGIN
    -- 引用元の情報を格納する変数を用意
    l_citation := '<table><tr><th>ファイル名</th><th>該当テキスト</th></tr>';
    
    -- ベクトル検索を実行し、RAGに使用するコンテキストを取得
    IF :P4_CATEGORY IS NULL THEN
        IF :P4_DOCNUM IS NULL THEN
            -- カテゴリ、保証ドキュメント数の指定がない場合のクエリ
            FOR rec IN (SELECT doc.doc_filename doc_filename, em.embed_data embed_data
                FROM documents doc, embed em 
                WHERE doc.id = em.doc_id AND TRUNC(doc.doc_lastupd) BETWEEN :P4_BEGIN_DATE AND :P4_END_DATE
                ORDER BY VECTOR_DISTANCE(em.embedding, TO_VECTOR(VECTOR_EMBEDDING(doc_model USING :P4_QUESTION as data)), COSINE)
                FETCH EXACT FIRST :P4_NUM ROWS ONLY)
            LOOP
                -- プロンプトに埋め込むコンテキストを追記
                l_context := l_context || rec.embed_data;
                -- 引用元情報を追記
                l_citation := l_citation || '<tr><td>' || rec.doc_filename || '</td><td>' || rec.embed_data || '</td></tr>';
            END LOOP;
        ELSE
            -- カテゴリの指定がなく、保証ドキュメント数の指定がある場合のクエリ
            FOR rec IN (SELECT doc.doc_filename doc_filename, em.embed_data embed_data
                FROM documents doc, embed em 
                WHERE doc.id = em.doc_id AND TRUNC(doc.doc_lastupd) BETWEEN :P4_BEGIN_DATE AND :P4_END_DATE
                ORDER BY VECTOR_DISTANCE(em.embedding, TO_VECTOR(VECTOR_EMBEDDING(doc_model USING :P4_QUESTION as data)), COSINE)
                FETCH EXACT FIRST :P4_DOCNUM PARTITIONS BY doc.id, :P4_NUM ROWS ONLY)
            LOOP
                -- プロンプトに埋め込むコンテキストを追記
                l_context := l_context || rec.embed_data;
                -- 引用元情報を追記
                l_citation := l_citation || '<tr><td>' || rec.doc_filename || '</td><td>' || rec.embed_data || '</td></tr>';
            END LOOP;
        END IF;
    ELSE
        -- カテゴリの指定があり、保証ドキュメント数の指定がない場合のクエリ
        IF :P4_DOCNUM IS NULL THEN
            FOR rec IN (SELECT doc.doc_filename doc_filename, em.embed_data embed_data
                FROM documents doc, embed em 
                WHERE doc.id = em.doc_id AND 
                doc.category IN (SELECT column_value FROM APEX_STRING.SPLIT(:P4_CATEGORY,':')) AND 
                TRUNC(doc.doc_lastupd) BETWEEN :P4_BEGIN_DATE AND :P4_END_DATE
                ORDER BY VECTOR_DISTANCE(em.embedding, TO_VECTOR(VECTOR_EMBEDDING(doc_model USING :P4_QUESTION as data)), COSINE)
                FETCH EXACT FIRST :P4_NUM ROWS ONLY)
            LOOP
                -- プロンプトに埋め込むコンテキストを追記
                l_context := l_context || rec.embed_data;
                -- 引用元情報を追記
                l_citation := l_citation || '<tr><td>' || rec.doc_filename || '</td><td>' || rec.embed_data || '</td></tr>';
            END LOOP;
        ELSE
            -- カテゴリ、保証ドキュメント数の指定がある場合のクエリ
            FOR rec IN (SELECT doc.doc_filename doc_filename, em.embed_data embed_data
                FROM documents doc, embed em 
                WHERE doc.id = em.doc_id AND 
                doc.category IN (SELECT column_value FROM APEX_STRING.SPLIT(:P4_CATEGORY,':')) AND 
                TRUNC(doc.doc_lastupd) BETWEEN :P4_BEGIN_DATE AND :P4_END_DATE
                ORDER BY VECTOR_DISTANCE(em.embedding, TO_VECTOR(VECTOR_EMBEDDING(doc_model USING :P4_QUESTION as data)), COSINE)
                FETCH EXACT FIRST :P4_DOCNUM PARTITIONS BY doc.id, :P4_NUM ROWS ONLY)
            LOOP
                -- プロンプトに埋め込むコンテキストを追記
                l_context := l_context || rec.embed_data;
                -- 引用元情報を追記
                l_citation := l_citation || '<tr><td>' || rec.doc_filename || '</td><td>' || rec.embed_data || '</td></tr>';
            END LOOP;
        END IF;
    END IF;
    
    -- プロンプト生成
    -- ベクトル検索結果をコンテキストとして埋め込むことでRAGを実装
    l_prompt := 'Answer the following question using the supplied context assuming you are a subject matter expert. Question: ' || :P4_QUESTION || ' Context: ' || l_context; 
    
    -- LLMにPOSTリクエストするためのBody設定
    l_body := '{ "message": "'|| APEX_ESCAPE.JSON(l_prompt) ||'" }';

    -- LLMにPOSTリクエストするためのHeader設定
    APEX_WEB_SERVICE.G_REQUEST_HEADERS.DELETE; 
    APEX_WEB_SERVICE.G_REQUEST_HEADERS(1).NAME  := 'Content-Type'; 
    APEX_WEB_SERVICE.G_REQUEST_HEADERS(1).VALUE := 'application/json';  

    -- LLMにREST API発行
    l_response_json := APEX_WEB_SERVICE.MAKE_REST_REQUEST 
    (p_url                  => l_cohere_rest_url, 
        p_http_method          => 'POST', 
        p_body                 => l_body, 
        p_credential_static_id => l_web_cred); 

    -- LLMからの応答を回答表示用アイテムに格納
    FOR row_1 IN C1 LOOP
        :P4_ANSWER := row_1.text; 
    END LOOP;
    
    -- 引用元情報を引用元表示用アイテムに格納
    :P4_CITATION := l_citation || '</table>';
END;

ONNXによるDB内Embedding方式特有の構文は 24行目 が該当します。

ORDER BY VECTOR_DISTANCE(em.embedding, TO_VECTOR(VECTOR_EMBEDDING(doc_model USING :P4_QUESTION as data)), COSINE)

VECTOR_EMBEDDING というSQL関数はONNXを利用する場合のみ使える関数です。
doc_model は事前にONNX import時に作成したONNX向けDBオブジェクトのオブジェクト名です。
USING 句の直後にEmbeddingしたいテキストを入れます。

詳細は下記マニュアルをご参照ください。
VECTOR_EMBEDDING

(参考) ベクトル索引の利用

生成AIからの応答速度は実運用する上で重要です。
特にRAGを使ったベクトル検索では、ベクトル検索の応答速度も留意が必要です。
23aiではベクトル検索を高速化するために、下記2つのベクトル索引を利用できます。

HNSW索引

HNSW索引の利用には vector_memory_size という新しい初期化パラメータの設定が必要です。
HNSW索引向けに、23aiからSGAに新しい領域「Vector Pool」が作られました。
詳細は下記マニュアルをご参照ください。
Size the Vector Pool

また23.4では、HNSW索引を利用する際の制限として、HNSW索引を作ったテーブルにDMLを実行できなくなります。
実行した場合 ORA-51928 が発生します。

まずは vector_memory_size を設定します。

-- sysユーザでCDBにログイン
SQL> show parameter vector_memory_size

NAME     TYPE
------------------------------------ ---------------------------------
VALUE
------------------------------
vector_memory_size     big integer
0

SQL> ALTER SYSTEM SET vector_memory_size=1G SCOPE=SPFILE;
SQL> shutdown immediate
SQL> startup

ORACLE

Total System Global Area 1603726344 bytes
Fixed Size    5360648 bytes
Variable Size  335544320 bytes
Database Buffers  184549376 bytes
Redo Buffers    4530176 bytes
Vector Memory Area 1073741824 bytes

SQL> show parameter vector_memory_size

NAME     TYPE
------------------------------------ ---------------------------------
VALUE
------------------------------
vector_memory_size     big integer
1G

HNSW索引を作成します。

-- vectorユーザで実行
SQL> CREATE VECTOR INDEX hnsw_idx ON embed (embedding)
ORGANIZATION INMEMORY NEIGHBOR GRAPH
DISTANCE COSINE
WITH TARGET ACCURACY 95;

Vector Pool の利用状況は以下のように V$VECTOR_MEMORY_POOL ビューで確認できます。

SQL> select CON_ID, POOL, ALLOC_BYTES/1024/1024 as ALLOC_BYTES_MB,
USED_BYTES/1024/1024 as USED_BYTES_MB
from V$VECTOR_MEMORY_POOL order by 1,2;

    CON_ID POOL                                                                           ALLOC_BYTES_MB USED_BYTES_MB
---------- ------------------------------------------------------------------------------ -------------- -------------
         3 1MB POOL                                                                                  704             1
         3 64KB POOL                                                                                 304         .3125
         3 IM POOL METADATA                                                                           16            16

1MB POOL がHNSW索引本体を格納する領域、64KB POOL がメタデータ情報を格納する領域です。
この構造はIn-Memoryと類似してますね。

実際にHNSW索引を使った問合せを実行し、実行計画を確認します。

SQL> SELECT embed_data
FROM embed
ORDER BY vector_distance(embedding, DBMS_VECTOR.UTL_TO_EMBEDDING('営業課題は何ですか?', JSON(:onnx_params)), COSINE)
FETCH APPROXIMATE FIRST 2 ROWS ONLY;

Plan hash value: 3880687600

-------------------------------------------------------------------------------------------
| Id  | Operation                      | Name     | Rows  | Bytes | Cost (%CPU)| Time     |
-------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT               |          |     2 |  4004 |     2  (50)| 00:00:01 |
|*  1 |  COUNT STOPKEY                 |          |       |       |            |          |
|   2 |   VIEW                         |          |     2 |  4004 |     2  (50)| 00:00:01 |
|*  3 |    SORT ORDER BY STOPKEY       |          |     2 |  9282 |     2  (50)| 00:00:01 |
|   4 |     TABLE ACCESS BY INDEX ROWID| EMBED    |     2 |  9282 |     1   (0)| 00:00:01 |
|   5 |      VECTOR INDEX HNSW SCAN    | HNSW_IDX |     2 |  9282 |     1   (0)| 00:00:01 |
-------------------------------------------------------------------------------------------

Predicate Information (identified by operation id):
---------------------------------------------------

   1 - filter(ROWNUM<=2)
   3 - filter(ROWNUM<=2)



----------------------------------------------------------
       6307  recursive calls
         46  db block gets
       8839  consistent gets
         12  physical reads
        452  redo size
       1572  bytes sent via SQL*Net to client
        108  bytes received via SQL*Net from client
          2  SQL*Net roundtrips to/from client
        405  sorts (memory)
          0  sorts (disk)
          2  rows processed

HNSW索引を使っている様子が実行計画に表れています。
ちなみに上位何件を取得するか、を指定する下記構文ですが、索引を使わない場合と異なる記法になっています。

-- 索引未利用時
FETCH EXACT FIRST :P4_NUM ROWS ONLY
-- 索引利用時
FETCH APPROXIMATE FIRST 2 ROWS ONLY;

具体的には EXACTAPPROXIMATE になっています。
索引利用時はSQLの書き方にご注意ください。
ちなみにHNSW索引、IVF索引ともにですが、索引は検索精度を若干下げる代わりに、応答速度を高速化する手法です。

またこの状態でAPEXアプリから新規ファイルを登録しようとすると、以下のように ORA-51928 が発生しエラーになります。
WS000091.JPG

IVF索引

IVF索引を利用するにあたり、初期化パラメータ等の設定は不要です。
以下のようにIVF索引を作成できます。

SQL> CREATE VECTOR INDEX ivf_idx ON embed (embedding) 
ORGANIZATION NEIGHBOR PARTITIONS 
DISTANCE COSINE 
WITH TARGET ACCURACY 95;

またIVF索引については、作成した表に対するDML実行の制限はありません。

以上がアプリ実装 後編でした。
本アプリの実装例が23aiを使ったRAGシステム構築の一助になりましたら幸いです。

5
4
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
5
4