Unnesting Node database calls

后端 未结 5 873
半阙折子戏
半阙折子戏 2021-01-19 03:27

I have an ordinary

var express = require(\'express\')

Node express www page, using session, pug, etc as usual. My db calls

var         


        
5条回答
  •  一向
    一向 (楼主)
    2021-01-19 03:32

    You can get rid of nested database calls by using promises.

    Since you mentioned that you are using mysql library for interacting with the database, unfortunately, this library doesn't provides a promise-based API. So to get rid of nested database calls in your code, you need to create a promise-based wrapper around the callback version of database calls.

    For a general overview of what promises are and how they work, see the following links:

    • MDN - Promise.
    • MDN - Using Promises

    Following is an example of how you can create a promise-based wrapper and then use that wrapper to get rid of nested database calls.

    This promise-based wrapper is just a function that returns a promise. It creates a promise instance, wraps the underlying database call and eventually when the database call returns the data, it notifies your code.

    function getCats() {
       return new Promise((resolve, reject) => {
           // make the database call
           db.cats((error, cats) => {
               // in case of an error, reject the promise by
               // calling "reject" function
               // Also pass the "error" object to the "reject" function
               // as an argument to get access to the error message 
               // in the code that calls this "getCats" function
               if (error) {
                  reject(error);
                  return;
               }
               
               // if there was no error, call "resolve" function
               // to resolve the promise. Promise will be resolved 
               // in case of successful database call
               // Also pass the data to "resolve" function
               // to access this data in the code that calls this
               // "getCats" function
               resolve(cats);
           });
       });
    }
    

    Now to in your route handler function, instead of calling db.cats(...), call this getCats wrapper function.

    There are two ways you can call the function that returns a promise:

    • Promise-chaining (For details, visit the links mentioned above)
    • async-await syntax (Recommended)

    Following code example uses async-await syntax. For this, first mark the route handler function as async by using the async keyword before the function keyword. Doing this, we can use await keyword inside this route handler function.

    app.get('/pets', async function(req, res, next) {
        try {
           const cats = await getCats();
           // similar wrappers for other database calls
           const dogs = await getDogs();
           const budgies = await getBudgies();
           
           // render the pub template, passing in the data
           // fetched from the database 
           ...
    
         catch (error) {
           // catch block will be invoked if the promise returned by
           // the promise-based wrapper function is rejected
           // handle the error appropriately
         }
    });
    

    Above code example only shows how to wrap the db.cats(...) database call in a promise-based wrapper and use that wrapper to get the data from the database. Similarly, you can create wrappers for db.dogs(...) and db.budgies(...) calls.

    Instead of creating a separate promise-based wrapper for each database call, ideally, you should create a re-usable promise-based wrapper function that takes in a function to call and wraps that function call in a promise just like shown in the above code example, i.e. getCats function.

    Parallel Database calls

    One important thing to note in the above code in the route handler function

    const cats = await getCats();
    const dogs = await getDogs();
    const budgies = await getBudgies();
    

    is that this will lead to sequential database calls which may or may not what you want.

    If these database calls do not depend on each other, then you can call the promise-based wrappers in parallel using Promise.all() method.

    Following code example shows how you can call your promise-based wrapper functions in parallel using Promise.all().

    app.get('/pets', async function(req, res, next) {
        try {
           // "petsData" will be an array that will contain all the data from 
           // three database calls.
           const petsData = await Promise.all([getCats(), getDogs(), getBudgies()]);
           
           // render the pub template, passing in the data
           // fetched from the database 
           ...
     
         catch (error) {
           ...
         }
     });
    

    I hope this is enough to help you get rid of the nested database calls in your current code and start using promises in your code.

提交回复
热议问题