Skip to content

Commit cca7118

Browse files
authored
Deferred: Rename getStackHook to getErrorHook (3.x version)
Rename `jQuery.Deferred.getStackHook` to `jQuery.Deferred.getErrorHook` to indicate passing an error instance is usually a better choice - it works with source maps while a raw stack generally does not. In jQuery `3.7.0`, we'll keep both names, marking the old one as deprecated. In jQuery `4.0.0` we'll just keep the new one. This change implements the `3.7.0` version; PR gh-5211 implements the `4.0.0` one. Fixes gh-5201 Closes gh-5212 Ref gh-5211
1 parent 3b7bf19 commit cca7118

File tree

4 files changed

+58
-11
lines changed

4 files changed

+58
-11
lines changed

src/deferred.js

+11-4
Original file line numberDiff line numberDiff line change
@@ -194,7 +194,7 @@ jQuery.extend( {
194194

195195
if ( jQuery.Deferred.exceptionHook ) {
196196
jQuery.Deferred.exceptionHook( e,
197-
process.stackTrace );
197+
process.error );
198198
}
199199

200200
// Support: Promises/A+ section 2.3.3.3.4.1
@@ -222,10 +222,17 @@ jQuery.extend( {
222222
process();
223223
} else {
224224

225-
// Call an optional hook to record the stack, in case of exception
225+
// Call an optional hook to record the error, in case of exception
226226
// since it's otherwise lost when execution goes async
227-
if ( jQuery.Deferred.getStackHook ) {
228-
process.stackTrace = jQuery.Deferred.getStackHook();
227+
if ( jQuery.Deferred.getErrorHook ) {
228+
process.error = jQuery.Deferred.getErrorHook();
229+
230+
// The deprecated alias of the above. While the name suggests
231+
// returning the stack, not an error instance, jQuery just passes
232+
// it directly to `console.warn` so both will work; an instance
233+
// just better cooperates with source maps.
234+
} else if ( jQuery.Deferred.getStackHook ) {
235+
process.error = jQuery.Deferred.getStackHook();
229236
}
230237
window.setTimeout( process );
231238
}

src/deferred/exceptionHook.js

+6-2
Original file line numberDiff line numberDiff line change
@@ -9,12 +9,16 @@ define( [
99
// warn about them ASAP rather than swallowing them by default.
1010
var rerrorNames = /^(Eval|Internal|Range|Reference|Syntax|Type|URI)Error$/;
1111

12-
jQuery.Deferred.exceptionHook = function( error, stack ) {
12+
// If `jQuery.Deferred.getErrorHook` is defined, `asyncError` is an error
13+
// captured before the async barrier to get the original error cause
14+
// which may otherwise be hidden.
15+
jQuery.Deferred.exceptionHook = function( error, asyncError ) {
1316

1417
// Support: IE 8 - 9 only
1518
// Console exists when dev tools are open, which can happen at any time
1619
if ( window.console && window.console.warn && error && rerrorNames.test( error.name ) ) {
17-
window.console.warn( "jQuery.Deferred exception: " + error.message, error.stack, stack );
20+
window.console.warn( "jQuery.Deferred exception: " + error.message,
21+
error.stack, asyncError );
1822
}
1923
};
2024

test/unit/deferred.js

+5-5
Original file line numberDiff line numberDiff line change
@@ -628,13 +628,13 @@ QUnit[ window.console ? "test" : "skip" ]( "jQuery.Deferred.exceptionHook with s
628628
defer = jQuery.Deferred(),
629629
oldWarn = window.console.warn;
630630

631-
jQuery.Deferred.getStackHook = function() {
631+
jQuery.Deferred.getErrorHook = function() {
632632

633633
// Default exceptionHook assumes the stack is in a form console.warn can log,
634-
// but a custom getStackHook+exceptionHook pair could save a raw form and
634+
// but a custom getErrorHook+exceptionHook pair could save a raw form and
635635
// format it to a string only when an exception actually occurs.
636636
// For the unit test we just ensure the plumbing works.
637-
return "NO STACK FOR YOU";
637+
return "NO ERROR FOR YOU";
638638
};
639639

640640
window.console.warn = function() {
@@ -656,13 +656,13 @@ QUnit[ window.console ? "test" : "skip" ]( "jQuery.Deferred.exceptionHook with s
656656
} else {
657657
assert.ok( /cough_up_hairball/.test( msg ), "Function mentioned: " + msg );
658658
}
659-
assert.ok( /NO STACK FOR YOU/.test( msg ), "Stack trace included: " + msg );
659+
assert.ok( /NO ERROR FOR YOU/.test( msg ), "Error included: " + msg );
660660
};
661661
defer.then( function() {
662662
jQuery.cough_up_hairball();
663663
} ).then( null, function( ) {
664664
window.console.warn = oldWarn;
665-
delete jQuery.Deferred.getStackHook;
665+
delete jQuery.Deferred.getErrorHook;
666666
done();
667667
} );
668668

test/unit/deprecated.js

+36
Original file line numberDiff line numberDiff line change
@@ -661,4 +661,40 @@ QUnit.test( "trim", function( assert ) {
661661
assert.equal( jQuery.trim( "\uFEFF \xA0! | \uFEFF" ), "! |", "leading/trailing should be trimmed" );
662662
} );
663663

664+
if ( includesModule( "deferred" ) ) {
665+
QUnit.test( "jQuery.Deferred.exceptionHook with stack hooks", function( assert ) {
666+
667+
assert.expect( 2 );
668+
669+
var done = assert.async(),
670+
defer = jQuery.Deferred(),
671+
oldWarn = window.console.warn;
672+
673+
jQuery.Deferred.getStackHook = function() {
674+
675+
// Default exceptionHook assumes the stack is in a form console.warn can log,
676+
// but a custom getStackHook+exceptionHook pair could save a raw form and
677+
// format it to a string only when an exception actually occurs.
678+
// For the unit test we just ensure the plumbing works.
679+
return "NO STACK FOR YOU";
680+
};
681+
682+
window.console.warn = function() {
683+
var msg = Array.prototype.join.call( arguments, " " );
684+
assert.ok( /cough_up_hairball/.test( msg ), "Function mentioned: " + msg );
685+
assert.ok( /NO STACK FOR YOU/.test( msg ), "Stack trace included: " + msg );
686+
};
687+
688+
defer.then( function() {
689+
jQuery.cough_up_hairball();
690+
} ).then( null, function() {
691+
window.console.warn = oldWarn;
692+
delete jQuery.Deferred.getStackHook;
693+
done();
694+
} );
695+
696+
defer.resolve();
697+
} );
698+
}
699+
664700
}

0 commit comments

Comments
 (0)