Skip to content

Commit

Permalink
Catch errors in async queue & re-start. Fixes #1933. Fixes #1759.
Browse files Browse the repository at this point in the history
  • Loading branch information
kevinpschaaf committed Jun 24, 2015
1 parent f62a80d commit 90caa2b
Show file tree
Hide file tree
Showing 2 changed files with 76 additions and 35 deletions.
70 changes: 36 additions & 34 deletions src/lib/async.html
Original file line number Diff line number Diff line change
Expand Up @@ -9,60 +9,62 @@
-->
<script>

Polymer.Async = (function() {

var currVal = 0;
var lastVal = 0;
var callbacks = [];
var twiddle = document.createTextNode('');
Polymer.Async = {

function runAsync(callback, waitTime) {
_currVal: 0,
_lastVal: 0,
_callbacks: [],
_twiddleContent: 0,
_twiddle: document.createTextNode(''),

run: function (callback, waitTime) {
if (waitTime > 0) {
return ~setTimeout(callback, waitTime);
} else {
twiddle.textContent = currVal++;
callbacks.push(callback);
return currVal - 1;
this._twiddle.textContent = this._twiddleContent++;
this._callbacks.push(callback);
return this._currVal++;
}
}
},

function cancelAsync(handle) {
cancel: function(handle) {
if (handle < 0) {
clearTimeout(~handle);
} else {
var idx = handle - lastVal;
var idx = handle - this._lastVal;
if (idx >= 0) {
if (!callbacks[idx]) {
if (!this._callbacks[idx]) {
throw 'invalid async handle: ' + handle;
}
callbacks[idx] = null;
this._callbacks[idx] = null;
}
}
}
},

function atEndOfMicrotask() {
var len = callbacks.length;
_atEndOfMicrotask: function() {
var len = this._callbacks.length;
for (var i=0; i<len; i++) {
var cb = callbacks[i];
var cb = this._callbacks[i];
if (cb) {
cb();
try {
cb();
} catch(e) {
// Clear queue up to this point & start over after throwing
i++;
this._callbacks.splice(0, i);
this._lastVal += i;
this._twiddle.textContent = this._twiddleContent++;
throw e;
}
}
}
callbacks.splice(0, len);
lastVal += len;
this._callbacks.splice(0, len);
this._lastVal += len;
}
};

new (window.MutationObserver || JsMutationObserver)(atEndOfMicrotask)
.observe(twiddle, {characterData: true})
;

// exports

return {
run: runAsync,
cancel: cancelAsync
};

})();
new (window.MutationObserver || JsMutationObserver)
(Polymer.Async._atEndOfMicrotask.bind(Polymer.Async))
.observe(Polymer.Async._twiddle, {characterData: true});

</script>
41 changes: 40 additions & 1 deletion test/unit/async.html
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@
<body>

<script>

setup(function() {
window.Async = Polymer.Async;
});
Expand Down Expand Up @@ -80,6 +80,45 @@
});
});

test('error handling', function(done) {
var called1 = 0;
var called2 = 0;
var called3 = 0;
var called4 = 0;
var callback1 = function() {
called1++;
};
var callback2 = function() {
called2++;
throw new Error('intentional error 2');
};
var callback3 = function() {
called3++;
throw new Error('intentional error 2');
};
var callback4 = function() {
called4++;
};
Async.run(callback1);
Async.run(callback2);
Async.run(callback3);
Async.run(callback4);
// Force synchronous microtask, since we can't catch the error otherwise
assert.throws(function(done) {
Async._atEndOfMicrotask();
});
assert.throws(function(done) {
Async._atEndOfMicrotask();
});
setTimeout(function() {
assert.equal(called1, 1);
assert.equal(called2, 1);
assert.equal(called3, 1);
assert.equal(called4, 1);
done();
});
});

});

suite('cancel no-wait async', function() {
Expand Down

0 comments on commit 90caa2b

Please sign in to comment.