web-dev-qa-db-ja.com

Dialogflowフルフィルメント内でのサードパーティAPIの使用

インラインエディター(Cloud Functions for Firebaseを使用)を使用しているDialogflowエージェントがあります。 HTTPS GETハンドラーをインテントハンドラーに埋め込もうとすると、「終了した関数からの例外を無視しています」というログエントリでクラッシュします。多分約束でこれを行うより良い方法があるかもしれませんが、私はそれが初めてです。実際には、Blazeプランへのアップグレード後に外部クエリを実行することがわかります。これは、請求先アカウントの制限ではありません。とにかく、ここにコードがあります:

'use strict';

const functions = require('firebase-functions');
//const rp = require('request-promise');
const {WebhookClient} = require('dialogflow-fulfillment');
const {Card, Suggestion} = require('dialogflow-fulfillment');

process.env.DEBUG = 'dialogflow:debug'; // enables lib debugging statements

exports.dialogflowFirebaseFulfillment = functions.https.onRequest((request, response) => {
  const agent = new WebhookClient({ request, response });
  console.log('Dialogflow Request headers: ' + JSON.stringify(request.headers));
  console.log('Dialogflow Request body: ' + JSON.stringify(request.body));

  function welcome(agent) {
    agent.add(`Welcome to my agent!`);
  }

  function fallback(agent) {
    agent.add(`I didn't understand`);
    agent.add(`I'm sorry, can you try again?`);
  }

  function findWidget(agent) {

    const https = require("https");
    const url = "https://api.site.com/sub?query=";

    agent.add(`Found a widget for you:`);

    var widgetName = getArgument('Name');

    https.get(url + widgetName, res => {
      res.setEncoding("utf8");
      let body = "";
      res.on("data", data => {
        body += data;
      });
      res.on("end", () => {

        body = JSON.parse(body);

        agent.add(new Card({
            title: `Widget ` + body.name,
            text: body.description,
            buttonText: 'Open link',
            buttonUrl: body.homepage
          })
        );
      });
    });  
  }

// Run the proper function handler based on the matched Dialogflow intent name
let intentMap = new Map();
intentMap.set('Default Welcome Intent', welcome);
intentMap.set('Default Fallback Intent', fallback);
intentMap.set('Find Old Movie Intent', findOldMovie);
intentMap.set('Find Movie Intent', findMovie);
// intentMap.set('your intent name here', googleAssistantHandler);
agent.handleRequest(intentMap);
});
6
Geoff Wong

Promiseでこれを行うには「より良い」方法があるだけではありません。非同期に呼び出されるコードがある場合、agent.handleRequest()はPromiseを使用する必要があります。問題は、https.getが実行を完了する前にfindWidget関数が返されるため、応答が送信されないことです。

私は request-promise-native パッケージを使用してHTTP呼び出しを行う傾向があります。これは、多くのことを簡素化するためです。 (ただし、Promiseで呼び出しを自由にラップできます。)したがって、Promiseバージョンは次のようになります(テストされていません)。

  var findWidget = function(agent) {

    const request = require('request-promise-native');
    const url = "https://api.site.com/sub?query=";

    agent.add(`Found a widget for you:`);

    var widgetName = getArgument('Name');

    return request.get( url+widgetName )
      .then( jsonBody => {
        var body = JSON.parse(jsonBody);
        agent.add(new Card({
          title: `Widget ` + body.name,
          text: body.description,
          buttonText: 'Open link',
          buttonUrl: body.homepage
        }));
        return Promise.resolve( agent ); 
      });
    });  
  }
8
Prisoner