HTTP request with Alexa and Lambda on SDK v2, how to make it works?
Asked Answered
V

3

6

I am playing with ASK SDK v2 provided by Amazon in order to make Skill for Alexa but I face an architectural problem :

First of all, the HTTP request works like a charm but I would like to return speach response if and only if my HTTP request is complete but I don't even know if it's possible because of the "handle" function that should return something (look at comments) :

const MyIntentHandler = {

  canHandle(handlerInput) {
    const request = handlerInput.requestEnvelope.request;
    return request.type === 'LaunchRequest' || (request.type === 'IntentRequest' && request.intent.name === 'MyIntent');
  },
  handle(handlerInput) {

    var options = {
      host: 'http://foo.com',
      port: 80,
      path: '/mypath',
      method: 'GET'
    };

    var req = http.request(options, function(result){
      result.on("end", function(){
        //I would like to return speak here like that :
        //return handlerInput.responseBuilder.speak("It works").withSimpleCard("MyTestApp", "It works").getResponse()
      })
    });
    req.end();
    //And I would like to remove this line to manage response in result.on("end", function(){}) above
    return handlerInput.responseBuilder.speak("It works").withSimpleCard("MyTestApp", "It works").getResponse();
  },
};

Any idea to deal with this ?

Viewable answered 1/5, 2018 at 13:4 Comment(2)
try putting your return inside your result.on("end", function(){...}).Unequal
try using a promised approach, e.g. npm request-promise.Aspect
V
7

I found the official way to make it :

1) Create a new funtion that manage http request and return a promise :

function httpGet(options) {
  return new Promise(((resolve, reject) => {

    const request = http.request(options, (response) => {
      response.setEncoding('utf8');
      let returnData = '';

      if (response.statusCode < 200 || response.statusCode >= 300) {
        return reject(new Error(`${response.statusCode}: ${response.req.getHeader('host')} ${response.req.path}`));
      }

      response.on('data', (chunk) => {
        returnData += chunk;
      });

      response.on('end', () => {
        resolve(JSON.parse(returnData));
      });

      response.on('error', (error) => {
        reject(error);
      });
    });
    
    request.on('error', function (error) {
      reject(error);
    });
    
    request.end();
  }));
}

2) In the Intent, return a promise in which you call your httpGet function :

const MyIntentHandler = {

  canHandle(handlerInput) {
    const request = handlerInput.requestEnvelope.request;
    return request.type === 'LaunchRequest' || (request.type === 'IntentRequest' && request.intent.name === 'MyIntent');
  },
  handle(handlerInput) {

    var options = {
      host: 'http://foo.com',
      port: 80,
      path: '/mypath',
      method: 'GET'
    };

    return new Promise((resolve, reject) => {
     httpGet(options).then((response) => {
       resolve(handlerInput.responseBuilder.speak("It is done.").getResponse());
     }).catch((error) => {
        resolve(handlerInput.responseBuilder.speak('Thor is not available at the moment. Please try again later or contact your administrator.')
        .getResponse());
      });
    });
  },
};

This is the way to do it properly. My example is base on alexa petmatch sample.

Viewable answered 2/5, 2018 at 18:16 Comment(0)
C
4

Also we can use request module to call API as follows

const SearchIntentHandler = {
  canHandle(handlerInput) {
    return (
      handlerInput.requestEnvelope.request.type === "IntentRequest" &&
      handlerInput.requestEnvelope.request.intent.name === "SearchIntent"
    );
  },
  handle(handlerInput) {
    const query = handlerInput.requestEnvelope.request.intent.slots.SearchQuery.value;
    return new Promise((resolve, reject) => {
      getSearchResults(query).then((response) => {
        resolve(handlerInput.responseBuilder.speak(response).getResponse());
      }).catch((error) => {
        resolve(handlerInput.responseBuilder.speak('This is not available at the moment.').getResponse());
      });
    });
  }
};


function getSearchResults(query){
  return new Promise((resolve, reject)=>{
    let options = { 
      method: 'POST',
      url: 'http://url.com',
      headers: {'Cache-Control': 'no-cache','Content-Type': 'application/x-www-form-urlencoded' },
      form: { text: query } 
    };
    request(options, function (error, response, body) {
      if (error) throw new Error(error);
      let data = body ? JSON.parse(body) : ""
      return resolve(data);
    });
  });
}
Confess answered 1/10, 2018 at 11:38 Comment(0)
P
1

An Async/Await Example

  1. Convert handle to an async function.
  2. Await your promise

Example:

const SomeIntentHandler = {
  canHandle(handlerInput) {...},
  async handle(handlerInput) {
    return handlerInput.responseBuilder.speak(await promise).getResponse();
  }
};
Proliferation answered 25/10, 2018 at 20:27 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.