The asynchronous behaviour of Twig.js relies on promises, in order to support both the synchronous and asynchronous behaviour there is an internal promise implementation that runs fully synchronous.
The internal implementation of promises does not use setTimeout
to run through the promise chain, but instead synchronously runs through the promise chain.
The different promise implementations can be mixed, synchronous behaviour however is no longer guaranteed as soon as the regular promise implementation is run.
Internal (synchronous) implementation
console.log('start');
Twig.Promise.resolve('1')
.then(function(v) {
console.log(v);
return '2';
})
.then(function(v) {
console.log(v);
});
console.log('stop');
/**
* Prints to the console:
* start
* 1
* 2
* stop
*/
Regular / native promises
Implementations such as the native promises or bluebird promises.
console.log('start');
Promise.resolve('1')
.then(function(v) {
console.log(v);
return '2';
})
.then(function(v) {
console.log(v);
});
console.log('stop');
/**
* Prints to the console:
* start
* stop
* 1
* 2
*/
Mixing promises
console.log('start');
Twig.Promise.resolve('1')
.then(function(v) {
console.log(v);
return Promise.resolve('2');
})
.then(function(v) {
console.log(v);
});
console.log('stop');
/**
* Prints to the console:
* start
* 1
* stop
* 2
*/
To preserve the correct order of execution there is an implemenation of Twig.forEach()
that waits any promises returned from the callback before executing the next iteration of the loop. If no promise is returned the next iteration is invoked immediately.
var arr = new Array(5);
Twig.async.forEach(arr, function(value, index) {
console.log(index);
if (index % 2 == 0)
return index;
return Promise.resolve(index);
})
.then(function() {
console.log('finished');
});
/**
* Prints to the console:
* 0
* 1
* 2
* 3
* 4
* finished
*/
The rendering mode of Twig.js internally is determined by the allowAsync
argument that can be passed into Twig.expression.parse
, Twig.logic.parse
, Twig.parse
and Twig.Template.render
. Detecting if at any point code runs asynchronously is explained in detecting asynchronous behaviour.
For the end user switching between synchronous and asynchronous is as simple as using a different method on the template instance.
Render template synchronously
var output = twig({
data: 'a {{value}}'
}).render({
value: 'test'
});
/**
* Prints to the console:
* a test
*/
Render template asynchronously
var template = twig({
data: 'a {{value}}'
}).renderAsync({
value: 'test'
})
.then(function(output) {
console.log(output);
});
/**
* Prints to the console:
* a test
*/
The pattern used to detect asynchronous behaviour is the same everywhere it is used and follows a simple pattern.
- Set a variable
isAsync = true
- Run the promise chain that might contain some asynchronous behaviour.
- As the last method in the promise chain set
isAsync = false
- Underneath the promise chain test whether
isAsync
istrue
This pattern works because the last method in the chain will be executed in the next run of the eventloop (setTimeout
/setImmediate
).
Synchronous promises only
var isAsync = true;
Twig.Promise.resolve()
.then(function() {
// We run our work in here such to allow for asynchronous work
// This example is fully synchronous
return 'hello world';
})
.then(function() {
isAsync = false;
});
if (isAsync)
console.log('method ran asynchronous');
console.log('method ran synchronous');
/**
* Prints to the console:
* method ran synchronous
*/
Mixed promises
var isAsync = true;
Twig.Promise.resolve()
.then(function() {
// We run our work in here such to allow for asynchronous work
return Promise.resolve('hello world');
})
.then(function() {
isAsync = false;
});
if (isAsync)
console.log('method ran asynchronous');
console.log('method ran synchronous');
/**
* Prints to the console:
* method ran asynchronous
*/