2
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

Databricksからインターネットに接続するエージェントツールの構築

Last updated at Posted at 2025-03-10

存在は知っていたのですが、動かせていませんでした。

AIエージェントツールをSlack、Googleカレンダー、またはHTTPリクエストを使用してAPIを使用する任意のサービスなどの外部アプリケーションに接続する方法を学びます。 エージェントは、外部に接続されたツールを使用して、タスクの自動化、メッセージの送信、およびサードパーティプラットフォームからのデータの取得を行うことができます。

この手順では、Unity Catalogが管理する接続としてHTTP接続を作成し、認証情報と共にアクセス先をポイントするようにします。

ただ、こちらを読んだ時に「エグレスコントロールなどでネットワークが疎通していればここで説明されている手順でなくても、インターネットに接続するツールは作れるのは?」と思いました。

例えば、以下のようなツールを作成してUntiy Catalogに登録します。ちなみに、https://icanhazdadjoke.com/ はお父さんのジョークをランダムに返してくれます。

CREATE OR REPLACE FUNCTION takaakiyayoi_catalog.tools.tell_dad_joke() 
RETURNS STRING 
LANGUAGE PYTHON AS $$

import requests

url = "https://icanhazdadjoke.com/"
headers = {"Accept": "application/json"}
response = requests.get(url, headers=headers)

if response.status_code == 200:
   joke_data = response.json()
   return joke_data['joke']
else:
   return f"Failed to retrieve joke. Status code: {response.status_code}"
$$

こんな感じでPlaygroundで冗談を教えてもらうことができます。

Screenshot 2025-03-10 at 14.56.18.png

では、なぜ接続を用いてエージェントツールを構築するのでしょうか。元のマニュアルに説明があります:

Unity Catalog Connection を使用する利点は次のとおりです。

  • 安全な認証情報管理: シークレットとトークンは Unity Catalog に安全に保管され、管理されるため、ユーザーに公開されることはありません。
  • きめ細かなアクセス制御: Unity Catalog では、 USE_CONNECTIONMANAGE_CONNECTION の特権を使用して、接続を使用または管理できるユーザーをきめ細かく制御できます。
  • ホスト固有のトークンの強制: トークンは、接続の作成時に指定された host_name に制限されるため、承認されていないホストでは使用できません。

つまり、認証情報やアクセス先、そして接続へのアクセス権を集中管理できるというところにメリットがあるということです。また、接続として登録されたアクセス先にはhttp_request関数でSQLからもアクセスできるようになります。

こちらのSlackのサンプルノートブックを動かします。

AIエージェントのSlackメッセージングツール

このノートブックでは、事前に定義されたSlackチャンネルにメッセージを送信するAIエージェントツールを作成する2つの方法を示します。

方法1: Unity Catalog関数を使用してツールを作成

  1. Unity Catalog接続オブジェクトを作成: 接続オブジェクトは、Slackに接続するために必要な認証情報を保存します。
  2. 前処理および後処理のUnity Catalog関数を定義: リクエストを前処理してSlackメッセージをフォーマットし、Slack API呼び出し後のステータスを確認するためにレスポンスを処理します。
  3. Unity Catalog関数ツールを作成: Slack APIにクエリを送信してメッセージを投稿するUnity Catalog関数を作成します。

方法2: エージェントコードでツールを定義

  1. Unity Catalog接続オブジェクトを作成: 接続オブジェクトは、Slackに接続するために必要な認証情報を保存します。
  2. Pythonでツールを定義: 入力テキストを処理し、Slack APIにクエリを送信し、返り値を処理するPython関数を定義します。

Unity Catalog関数ツールとエージェントコードで定義されたツールの違いについては、Databricksのドキュメントを参照してください (AWS | Azure)。

%pip install -U databricks-sdk
%restart_python

UC接続の作成

このコードはUC接続オブジェクトを作成します。接続オブジェクトは、外部関数に認証するためのbearer_tokenを保存します。Slackのトークンの取得方法はこちらをご覧ください。

%sql
DROP CONNECTION IF EXISTS test_sql_slack_connection;
CREATE CONNECTION test_sql_slack_connection
TYPE HTTP
OPTIONS (
  host 'https://slack.com',
  port '443',
  base_path '/',
  bearer_token 'xoxb-...'
);

これでSlackへの接続が作成されました。

Screenshot 2025-03-10 at 15.10.26.png

方法1: Unity Catalog関数を使用してツールを作成

前処理関数

外部API呼び出しの前に呼び出される前処理関数を定義します。前処理関数は、Slackチャンネルとテキストを含む構造体を受け取り、テキストに免責事項メッセージを追加して修正された構造体を返します。

%sql
CREATE OR REPLACE FUNCTION takaakiyayoi_catalog.agents.pre_process_slack_message(data Struct < channel: STRING, text: STRING >) 
RETURNS Struct < channel: STRING, text: STRING > 
LANGUAGE PYTHON 
AS $$

# このメッセージはAIエージェントからのものですという免責事項を追加
slack_text = "このメッセージはAIエージェントからです:\n" + data.text

# チャンネル名とクリーンなメッセージを希望の形式で結合
return {"channel": data.channel, "text": data.text}
$$

後処理関数

外部API呼び出し後に呼び出される後処理関数を定義します。後処理関数は、ステータスコードとテキストとしてレスポンスデータを受け取り、文字列を返すことを期待します。この場合、関数はステータスコードを確認し、成功または失敗のメッセージを返します。

%sql
CREATE OR REPLACE FUNCTION takaakiyayoi_catalog.agents.post_process_slack_message(
  response_data Struct < statusCode: INT, text: STRING >
) 
RETURNS STRING 
LANGUAGE PYTHON AS $$

# 処理結果
if response_data.statusCode == 200:
    return "Slackへのメッセージ送信に成功しました!"
else:
    return "リクエスト実行中にエラーに遭遇しました。呼び出しからのメッセージ: " + text
$$

エージェントツールの定義

このコードは、テキストパラメータを受け取り、Slackチャンネルにメッセージを送信するUnity Catalog関数を定義します。関数は、接続オブジェクト、HTTPメソッド、リクエストボディ、および外部呼び出しのJSONペイロードを受け取るhttp_requestを使用します。

この例では、SlackチャンネルIDをハードコーディングしています。

%sql
CREATE OR REPLACE FUNCTION takaakiyayoi_catalog.agents.slack_send_message(text STRING COMMENT 'メッセージ内容')
RETURNS STRING
COMMENT "事前定義されたSlackチャンネル(ID: 'C085QCA8UKF'、名前: 'slack-integration-test')にメッセージを送信します。自動的に現在のユーザー名を先頭に追加します。入力: 送信するテキストメッセージ。戻り値: Slack APIの応答。使用例: SELECT main.default.slack_send_message('ここにメッセージを入力')."
RETURN (
  takaakiyayoi_catalog.agents.post_process_slack_message(
    http_request(
      conn => 'test_sql_slack_connection',
      method => 'POST',
      path => '/api/chat.postMessage',
      json =>
        to_json(
          takaakiyayoi_catalog.agents.pre_process_slack_message(
            named_struct('channel', "C085QCA8UKF", 'text', concat("On behalf of ", current_user(), ": ", text))
          )
        )
    )
  )
)

これでPlaygroundから動作確認することができます。

Screenshot 2025-03-10 at 15.12.57.png

Screenshot 2025-03-10 at 15.14.07.png

方法2: エージェントコードでツールを定義する

Python SDKを用いることで、同じ挙動をするエージェントを作成することができます。

from databricks.sdk import WorkspaceClient
from databricks.sdk.service.serving import ExternalFunctionRequestHttpMethod

def send_slack_message(text: str):
  """
    Slackチャンネル 'C085QCA8UKF' にメッセージを送信します。メッセージには自動的に現在のユーザー名が追加されます。戻り値: Slack APIのレスポンス。

    引数:
        text: Slackチャンネル 'channel-name' に投稿するテキストメッセージ。
  """

  slack_text = "This message is from an AI Agent:\n" + text

  response = WorkspaceClient().serving_endpoints.http_request(
    conn="test_sql_slack_connection",
    method=ExternalFunctionRequestHttpMethod.POST,
    path="/api/chat.postMessage",
    json={"channel": "C085QCA8UKF", "text": slack_text},
  )

  print(response.text)
  print(response.status_code)

  if response.status_code == 200:
    return "Successfully sent message to Slack!"
  else:
    return "Encountered failure when executing request. Message from Call: " + text
send_slack_message("Hello, world!")

Screenshot 2025-03-10 at 15.15.26.png

はじめてのDatabricks

はじめてのDatabricks

Databricks無料トライアル

Databricks無料トライアル

2
1
3

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
2
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?