Bot Framework Composer でボット開発 : その 2 ダイアログとトリガー


前回 は Bot Framework Composer のインストールを行いまいした。今回から実際のボットを作っていきます。
この記事ではボットの中心機能となるダイアログとトリガーを紹介します。

ダイアログ

QnA のような一問一答形式の場合、ボットはユーザーとの会話を記憶する必要はありませんが、複数回のやり取りを行う場合、どこまでやり取りしたか、またユーザーの回答が何であったかなどをコンテキストとして記憶しておく必要があります。

ダイアログはこのような会話の単位であり、簡単なボットであれば数個のダイアログ、複雑なボットであれば数十のダイアログから構成されます。

ダイアログの実行には以下の要素が必要です。

  • ユーザー入力の理解
  • トリガー
  • アクションの実行
  • ユーザー入力の保存
  • 応答の作成

ユーザー入力の理解

ユーザーがボットに話しかけた際、どのダイアログを実行するべきか判断する必要があります。Bot Framework Composer では数種類の方法をサポートしていますが、今回は単純なキーワードを使います。より高度な方法は以降の記事で紹介します。

トリガー

ダイアログを実行するきっかけとなるイベントをトリガーと呼び、ユーザー入力以外にも、ユーザーがボットを会話に追加した時や会話から退出した時、他のダイアログから呼び出しされた時などのシステムイベントがあります。

  • インテントトリガー: 自然言語解析を設定した場合に、ユーザーの意図解析の結果でトリガーする
  • ダイアログイベント: ダイアログの開始やキャンセル、エラーなどのイベントでトリガーする
  • アクティビティ: 会話の開始や終了、タイピング中などのシステムイベントでトリガーする
  • メッセージ: メッセージが届いた時や更新された時などのイベントでトリガーする
  • カスタムイベント: 独自に生成したイベントでトリガーする

アクションの実行

ユーザーに質問をしたり、HTTP リクエストを実行したりと、ダイアログの中で実行するアクションを設定します。Bot Framework Composer では非常に多くのアクションをサポートしますが、詳細は以降の記事で紹介し、今回はシンプルなアクションのみ使います。

ユーザー入力の保存

ユーザーに名前を聞いたり、年齢を聞いた場合、その結果を保存しておく必要があります。また状況に応じて保存した値を削除するなどの操作もサポートされます。

応答の作成

言語生成の機能を使い、動的な回答を作成したり、複数パターンから応答をランダムに返すことができる他、テキスト以外のリッチなカードを返すこともできます。今回はシンプルにテキスト応答のみ使います。

ボットのメモリー

ボットが会話を記憶する際、以下 2 つのオブジェクトにデータを格納します。

  • turn: ユーザーからのメッセージが来て、ボットが応答するまでの一度の会話のデータ
  • dialog: アクティブなダイアログとその親や子ダイアログのデータ

また以下 2 つのスコープがあります。

  • user: ユーザー単位の情報
  • conversation: 特定の会話単位の情報で会話が終了すると削除される

既存のボットから確認する

前回作成したオウム返しボットを見ていきます。

メインダイアログ

ボットを作成すると、指定した名前でメインのダイアログが作成されます。また既定で以下 2 つのトリガーが作成されます。

  • Unknown intent: インテントトリガーの一種で、意図が不明な場合にトリガーする
  • Greeting: アクティビティトリガーの一種で、ユーザーが会話に参加したなどのタイミングでトリガーする

Unknown intent
前回の記事で内容を確認した通り、ユーザーにオウム返しを行っています。

Greeting
Conversation Update はユーザーかボットの追加または会話からの退出でトリガーされます。ボットからの情報は turn というオブジェクトに格納されており、ここでは turn.Activity.members.Added で参加したメンバーのリストを取得しています。

foreach ループを使った場合、それぞれの値は dialog.foreach.value に含まれるため、MemberAdded のユーザーと、メッセージの宛先 (Recepient) が同じか if/else で確認しています。

では実際のデータの中身を見てみましょう。

1. Greeting の直後の「+」をクリックして、「Send a resposne」を追加。

2. 中身に - '${turn}' を指定。

3.「Test in Emulator」をクリックしてエミュレーターで検証。

4. 内容が確認できる。以下に turn.Activity.membersAdded と主なプロパティだけを抜粋。2 つのアカウント追加されたことが分かる。一方は "Bot" でもう一方は "User" だがこれらはあくまで表示名。

{
    "activity": {
        "type": "conversationUpdate",
        ...
        "recipient": {
            "id": "1ee85b90-6df9-11ea-8a1e-3be055026012",
            "name": "Bot",
            "role": "bot"
        },
        "membersAdded": [
            {
                "id": "1ee85b90-6df9-11ea-8a1e-3be055026012",
                "name": "Bot"
            },
            {
                "id": "e013a91c-c374-4232-bf64-fbb46de88cd7",
                "name": "User"
            }
        ],
        "membersRemoved": [],
        ...
        }
    }
}

Greeting の変更

現状 Greeting ではなにも処理されないため、内容を変更しましょう。

1. 追加した「Send a response」を削除。True ブランチに「Send a response」を追加。

2. 内容を - '${dialog.foreach.value.name}' さん、こんにちは。 に変更。

3.「Restart Bot」実行後、エミュレーターで動作を確認。

ダイアログを追加

最後にダイアログを追加してみましょう。

1.「New Dialog」をクリックして、ダイアログを追加。名前を「Time」にして次へ。

2. 作成した「Time」ダイアログを選択。「Begin Dialog」トリガーが追加されていることを確認。

3. Begin Dialog トリガーを選択し、「Send a response」を追加。内容に - 今の時間は '${addHours(utcNow() ,9)}'です。 と入力。

4. メインダイアログより「Unknown intent」を選択。アクションの追加より「Begin a new dialog」を選択。

5. Time ダイアログを選択。

6.「Restart Bot」クリック後、エミュレーターで検証。

まとめ

今回はダイアログとトリガーについて、簡単に見ていきました。次回はユーザーインプットの理解について見ていきます。

次の記事へ
目次へ戻る