Breaking the async boundary

James Allardice

The event loop

  • Used by JS engines to schedule tasks
  • Most callbacks (e.g. setTimeout) are tasks
  • Tasks are scheduled to run sequentially
  • Different tasks have different stack frames

The problem

startTimer();
a();
setTimeout(b, 0);
c();
stopTimer();

How long does it take the following program, including `b`, to execute?

startTimer();
a();
setTimeout(b, 0); // asynchronous 
c();
stopTimer();
startTimer();                                // 1
a();
setTimeout(b, 0); // asynchronous 
c();
stopTimer();
startTimer();                                // 1
a();                                         // 2
setTimeout(b, 0); // asynchronous 
c();
stopTimer();
startTimer();                                // 1
a();                                         // 2
setTimeout(b, 0); // asynchronous 
c();                                         // 3
stopTimer();
startTimer();                                // 1
a();                                         // 2
setTimeout(b, 0); // asynchronous 
c();                                         // 3
stopTimer();                                 // 4
startTimer();                                // 1
a();                                         // 2
setTimeout(b, 0); // asynchronous            // 5
c();                                         // 3
stopTimer();                                 // 4

The problem

How can we associate data with the HTTP request in the following program?

// Assume `app` is an Express instance
app.use((req, res, next) => {
  
  // Generate a unique identifier for this request.
  const requestId = generateId();
  next();
});
// Assume `app` is an Express instance
app.use((req, res, next) => {
  
  // Generate a unique identifier for this request.
  const requestId = generateId();

  // Attach the identifier to the request object.
  req.id = requestId;

  next();
});

The problem

How can we capture a full stack trace when the following program dies?

Solutions

  • Domains
  • Async Listener
  • Continuation Local Storage (CLS)
  • AsyncWrap
  • Zones

Domains

  • Introduced in 0.8 but now deprecated (not yet removed)
  • "Handle multiple different IO operations as a single group"

Async Listener

  • Pull request to Node core never merged
  • Polyfill exists
  • Provides a mechanism to run functions just before and after a new task begins and ends

Continuation Local Storage

  • Built on Async Listener
  • Associate values with a namespace
  • Namespaces are groups of functions, synchronous or otherwise
  • Retrieve value from a namespace in any function in that group

AsyncWrap

  • Undocumented API
  • Available via process.binding('async_wrap')

Zones

  • Stage 0 ECMAScript proposal
  • Inspired by Dart
  • Numerous existing implementations (Angular zone.js, StrongLoop zone, can-zone)
  • All tasks scheduled within a zone will execute within that context
  • Provides hooks into the event loop

Zones

// Requires Node 0.12 or earlier.

require('zone').enable();

zone
  .create(function () {

    startTimer();

    a();
    setTimeout(b, 0);
    c();
  })
  .setCallback(function () {
    stopTimer();
  });

Using StrongLoop zone which is no longer maintained and requires Node.js 0.12 or earlier.

Thank you for listening!

Any questions?

Breaking the async boundary

By James Allardice

Breaking the async boundary

An exploration of the problems caused by ECMAScript's event loop and some of the possible solutions to those problems.

  • 1,199