0

我必须在这里遗漏一些非常简单的东西,但是这里有。

我刚刚开始学习 Alexa 开发,并发现了 alexa-app 模块,它似乎使 Alexa 编程非常简单——除了我遇到的这个网络问题。

我正在浏览名为 AirportInfo 的团队提供的示例应用程序。问题区域的代码如下:

var faaHelper = new FAADataHelper();
faaHelper.requestAirportStatus(airportCode).then(function(airportStatus) {
    var goodMessage = faaHelper.formatAirportStatus(airportStatus);
    console.log(goodMessage);
    res.say(goodMessage).send();
}).catch(function(err) {
    console.log(err.statusCode);
    var prompt = 'I didn\'t have data for an airport code of ' + airportCode;
    console.log(prompt);
    res.say(prompt).reprompt(reprompt).shouldEndSession(false).send();
});
return false;

依次调用这些函数:

FAADataHelper.prototype.requestAirportStatus = function(airportCode) {
    return this.getAirportStatus(airportCode).then(
        function(response) {
            console.log('success - received airport info for ' + airportCode);
            return response.body;
        }
    );
};
FAADataHelper.prototype.getAirportStatus = function(airportCode) {
    var options = {
        method: 'GET',
        uri: ENDPOINT + airportCode,
        resolveWithFullResponse: true,
        json: true
    };
    return rp(options);
};

这对我来说看起来不错,但是当代码运行时,控制何时将响应发送回设备的主要 Alexa 应用“请求”比预期的要早返回。不是带有所选机场预期天气信息的完整响应负载,而是在return rp(options)呼叫后立即发回响应。.then()第一个代码示例中的块中执行的代码技能已经将相当于空响应的内容发送回 Alexa 之后运行。这实际上使 Alexa 崩溃,因为她说出了某种关于技能错误的神秘信息。

这是我的 server.js 代码:

var AlexaAppServer = require("../index.js");
AlexaAppServer.start({
    server_root: './',
    port: 8080,
    debug: true,
    // Use preRequest to load user data on each request and add it to the request json.
    // In reality, this data would come from a db or files, etc.
    preRequest: function(json, req, res) {
        console.log("preRequest fired");
        json.userDetails = { "name": "Bob Smith" };
    },
    // Add a dummy attribute to the response
    postRequest: function(json, req, res) {
        // look for this output in the log below
        console.log("postRequest fired");
        json.dummy = "text";
    }
});

这是显示我正在描述的这种情况的调试日志:

preRequest fired
REQUEST { method: 'GET',
  uri: 'http://services.faa.gov/airport/status/dfw',
  resolveWithFullResponse: true,
  json: true,
  simple: false,
  callback: [Function: RP$callback],
  transform: undefined,
  transform2xxOnly: false }
postRequest fired
REQUEST make request http://services.faa.gov/airport/status/dfw
REQUEST onRequestResponse http://services.faa.gov/airport/status/dfw 200 { date: 'Fri, 24 Mar 2017 05:09:41 GMT',
  server: 'Apache',

  ...

  'access-control-allow-origin': '*',
  'access-control-allow-methods': 'GET, HEAD, OPTIONS',
  'version-requested': 'Any',
  connection: 'close',
  'transfer-encoding': 'chunked',
  'content-type': 'application/json;charset=UTF-8' }
REQUEST reading response's body

...

REQUEST end event http://services.faa.gov/airport/status/dfw
REQUEST has body http://services.faa.gov/airport/status/dfw 517
REQUEST emitting complete http://services.faa.gov/airport/status/dfw
success - received airport info for dfw

注意相对于 的位置preRequest FiredpostRequest Fired显示success - received airport info for dfw。任何想法我做错了什么?在我的 Node 环境中配置错误,或者可能是错误的依赖版本?我很怀疑,因为它在我的调试器(VS Code)中从节点命令提示符和从 Lambda 中失败。

完整源代码链接:https ://github.com/bignerdranch/alexa-airportinfo

4

1 回答 1

4

我不了解有关 Alexa 的 Promises 的所有内容,但我确实明白了这一点。从您提出问题之日起,有更新版本的 alexa-app-server (3.0.1) 和 alexa-app (4.0.0) 可用。我最近使用这些版本开发了一项技能。alexa-app 2,alexa-airportinfo 中提到的版本,不适用于最新的 alexa-app-server。

所以,首先要做的是在你的代码中使用最新的 alexa-app。然后,意识到通过在 FAADataHelper 中使用 request-promise,代码将 Promise 返回给 index.js。为此,您需要返回该 Promise,而不是函数末尾的 false。

总而言之,我得到了最新的 alexa-app-server,将 airportinfo 中的 alexa-app 版本更改为最新版本,然后运行它,得到与您相同的结果 - 在 rp 请求完成之前返回响应。当我将代码更改为下面时,我得到了想要的结果 - 请求完成,然后响应完成,语音进入。

app.intent('airportinfo', {
  'slots': {
    'AIRPORTCODE': 'FAACODES'
  },
  'utterances': ['{|flight|airport} {|delay|status} {|info} {|for} {-|AIRPORTCODE}']
},
  function(req, res) {
    //get the slot
    var airportCode = req.slot('AIRPORTCODE');
    var reprompt = 'Tell me an airport code to get delay information.';
    if (_.isEmpty(airportCode)) {
      var prompt = 'I didn\'t hear an airport code. Tell me an airport code.';
      res.say(prompt).reprompt(reprompt).shouldEndSession(false);
      return true;
    } else {
      var faaHelper = new FAADataHelper();
      return faaHelper.requestAirportStatus(airportCode).then(function(airportStatus) {
        console.log(airportStatus);
        res.say(faaHelper.formatAirportStatus(airportStatus)).send();
      }).catch(function(err) {
        console.log(err.statusCode);
        var prompt = 'I didn\'t have data for an airport code of ' + airportCode;
         //https://github.com/matt-kruse/alexa-app/blob/master/index.js#L171
        res.say(prompt).reprompt(reprompt).shouldEndSession(false).send();
      });
      // return false;
    }
  }
);
于 2017-04-11T20:35:00.800 回答