Skip to content

Latest commit

 

History

History
184 lines (141 loc) · 6.88 KB

async-calls.md

File metadata and controls

184 lines (141 loc) · 6.88 KB

Frontend - Async calls (Ajax)

Javascript deals well with asynchronous calls like network requests. Historically, a reply was delivered though callback functions:

request(arguments, function callback(error, result) {
  // do something with result
});

But this may lead to spaghetti code therefore Promises were introduced:

var resultPromise = request(arguments);
resultPromise.then(function callback(result) {
  // Promise fulfilled, do something with result
});
resultPromise.catch(function callback(error) {
  // handle exceptions
});

The await keyword pauses asynchronous function until Promise is fulfilled.

try {
  var resultPromise = await request(arguments);
} catch (error) {

}

The promises can be integrated in the model in order to handle HTTP requests. This is possible through fetch method:

class Model extends Observable {
  async fetchImages() {
    const response = await fetchClient('/api/images').catch((error) => this.handleErrors(error));
    const images = await response.json();
    this.setImages(images);
  }

  setImages(images) {
    this.images = images;
    this.notify();
  }

  handleErrors(error) {
    this.error = error;
    this.notify();
  }
}

The fetchClient method is part of the framework, creates an HTTP request and returns a promise. Either await keyword or then method can be used to handle fulfilled promise.

Error can be handled within a try, catch blocks or using catch method of the promise object. It is a recommended to use a generic error handler. fetchImages also returns a promise because it has async keyword.

Because the web interface is not blocked when the method is paused with await keyword, the user can call it many times leading to unnecessary network usage. Repeated requests to the same resource can be avoid by adding a new state in the model, fetchingImages boolean in the following example does this:

class Model extends Observable {
  async fetchImages() {
    if (this.fetchingImages) {
      return;
    }

    this.fetchingImages = true;
    this.notify();

    const response = await fetchClient('/api/images').catch((error) => this.handleErrors(error));
    const images = await response.json();
    this.setImages(images);

    this.fetchingImages = false;
    this.notify();
  }

  setImages(images) {
    this.images = images;
    this.notify();
  }

  handleErrors(error) {
    this.error = error;
    this.notify();
  }
}

function button(model) {
  const action = () => model.fetchImages();
  const disabled = model.fetchingImages;

  // clicks are not handled when a button is 'disabled'
  return h('button', {onclick: action, disabled: disabled}, 'Fetch images')
}

fetchClient

fetchClient inherit from native fetch and automatically add a token to the request. All options from fetch can be used:

const options = {
  method: 'POST',
  headers: {
    'Accept': 'application/json',
    'Content-Type': 'application/json'
  },
  body: JSON.stringify({...})
};
const response = await fetchClient('/api/lock', options);

RemoteData

As a utility, the framework provides RemoteData type to encapsulate the different states of a data coming from the server. It may be used to deal with all possibilities like: are data asked? are they loaded yet? what if an error happen?

import {RemoteData} from '/js/src/index.js';

// A function that should display data from server (for example the name of the user) must handle every possibility
// Keep in mind that, at any time, the remote data will only match one of these possibilities
const getWelcomeMessage = (remoteData) => remoteData.match({
  NotAsked: () => 'Data has not been fetched from the server', // Display a message to the user saying that data has not been fetched
  Loading: () => 'Loading, please wait', // A request has probably been sent to the server but we did not receive any response yet
  Success: (name) => `Hello ${name}`, // The server response has been stored in the remote data payload, and it is passed as parameter to the Success callback
  Failure: (error) => `An error has occurred: ${error.message}`, // An error has occurred, displays its message
})

// When starting the application, the data may be not asked
let userNameRemoteData = RemoteData.notAsked();
getWelcomeMessage(userNameRemoteData); // Yields `Data has not been fetched from the server`

// At some point in the code, we fetch the data. We first set the remote data state to loading, then re-render (in real case, re-render is automated)
userNameRemoteData = RemoteData.loading();
// Update the message:
getWelcomeMessage(userNameRemoteData); // Yields `Loading, please wait`

// Suppose we have a function that resolves with the current user name fetched from the server
try {
  const username = await fetchUsernameFromServer();
  userNameRemoteData = RemoteData.success(username); // Everything gone right: store the username in a success remote data
} catch (error) {
  userNameRemoteData = RemoteData.failure(error); // An error occured, store it in a failure remote data
}
getWelcomeMessage() // And re-render!

// Sometimes, we need to execute some code only in case of successful remote data, in this case we can use the short branching of `match`
// For example, let's consider that we receive the total items count every time we fetch a page. We want to update the displayed pagination to update only if we actually receive a new pagination info, but keep the previous one in any case:
totalItems = remoteData.match({
  Success: ({newTotalItems}) => newTotalItems, // Use the new value if we have a success
  Other: () => totalItems // In any other cases, keep the previous value
})

This pattern uses tagged union type and is explained here: http://blog.jenkster.com/2016/06/how-elm-slays-a-ui-antipattern.html it is also comparable to solving the "accessing data from null pointer" problem.

Loader

The Loader model is an other utility to watch and simplify requests made with fetchClient, his goal is to be a "network manager".

import {Loader} from '/js/src/index.js';

const loader = new Loader();
const {result, ok} = await loader.post('/api/foo', {bar: 123, baz: 456})

The syntax is much more compact than using directly fetchClient.

  • result is the JSON decoded
  • ok is true on 2XX response only
  • loader.active is true if there is one or more requests working (used for a global spinner eventually)

In case of errors, the user can select between a prebuilt error message (which includes the status and statusText) by passing a third argument to the get(url, query, originalMessage) or post(url, options, originalMessage) methods

e.g. For a 404 error Key not found the messages will be sent as below:

  • originalMessage === true the error message will be Key not found
  • originalMessage === false the error message will be Request to server failed (404 Not Found): Key not found