-
Notifications
You must be signed in to change notification settings - Fork 27.4k
feat(minErr): set max depth for angular error JSON stringify #15433
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
On a high level it looks good. Still missing:
- Tests.
- Making the max depth configurable.
src/minErr.js
Outdated
@@ -57,7 +58,7 @@ function minErr(module, ErrorConstructor) { | |||
|
|||
for (i = SKIP_INDEXES, paramPrefix = '?'; i < templateArgs.length; i++, paramPrefix = '&') { | |||
message += paramPrefix + 'p' + (i - SKIP_INDEXES) + '=' + | |||
encodeURIComponent(toDebugString(templateArgs[i])); | |||
encodeURIComponent(toDebugString(templateArgs[i]), MAX_DEPTH); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
MAX_DEPTH
should be passed to toDebugString
, not encodeURIComponent
.
src/Angular.js
Outdated
var stackSource = []; | ||
var stackDest = []; | ||
var currentDepth = 0; | ||
maxDepth = (maxDepth && isNumber(maxDepth) && maxDepth > 0) ? maxDepth : -1; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I would simplify this: if (!isNumber(maxDepth)) maxDepth = NaN;
src/Angular.js
Outdated
|
||
function copyRecurse(source, destination, currentDepth) { | ||
currentDepth++; | ||
if (maxDepth !== -1 && currentDepth > maxDepth) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
By setting maxDepth
to NaN
when not specified, this condition could be: if (currentDepth > maxDepth)
src/stringify.js
Outdated
copy(obj, tempObj, maxDepth) | ||
} else { | ||
tempObj = obj; | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This could be simplified to:
if (isNumber(maxDepth)) obj = copy(obj, {}, maxDepth);
src/Angular.js
Outdated
function copyRecurse(source, destination, currentDepth) { | ||
currentDepth++; | ||
if (maxDepth !== -1 && currentDepth > maxDepth) { | ||
return typeof source; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I would prefer some less ambiguous value that also makes it clear there was more stuff that has been truncated. Also, typeof
can be misleading or at least not so useful, because almost anything that is not a primitive will return object
.
If we wanted to keep it simple, maybe something like '...'
. If we wanted to get more fancy, trying to get the name of the value's constructor and returning something like constructorName + '(...)'
would be nice, but probably not worth it.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
we are already using ...
for cyclic references so I think that we should do the same for this.
src/minErr.js
Outdated
@@ -34,6 +34,7 @@ function minErr(module, ErrorConstructor) { | |||
ErrorConstructor = ErrorConstructor || Error; | |||
return function() { | |||
var SKIP_INDEXES = 2; | |||
var MAX_DEPTH = 1; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There should be a way to customize this (I am not sure what it should be) and the default should be a little higher imo (maybe 3 or 4).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think to make max depth configurable we can,
var exampleMinErr = minErr('example');
throw exampleMinErr('one', 'This {0} is {1}', foo, bar);
- Refactor exampleMinErr to be
throw exampleMinErr({code:'one', message: 'This {0} is {1}', messageArgs:[foo, boo]});
and we can keep both styles working.
- Include the depth as a special string in the message like {$maxDepth=3}
'This {0} is {1} {$maxDepth=3}' so we get the depth
throw exampleMinErr('one', 'This {0} is {1} {$maxDepth=3}', foo, bar);
this Depth will be applied to all the message.
Or Include a depth for each placeholder like 'This {0:d8} is {1:d8} }'
throw exampleMinErr('one', 'This {0:d8} is {1:d8}', foo, bar);
d8 means that the depth for stringifying this object is 8.
I think the second solution is easier for implementation and have no backward compatibility Issue.
@gkalpak What is your opinion about that ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think it'll be useful in this case. See https://github.com/angular/angular.js/pull/15433/files#r97893306.
Wouldn't it be simpler and more performant just to provide a specialized |
I couldn't think of a way to track depth from |
Yeah, fair enough. |
fa1823e
to
d53eca0
Compare
d53eca0
to
3ae6f6b
Compare
@gkalpak var exampleMinErr = minErr('example');
throw exampleMinErr('one', 'This {0,maxDepth=2}', foo); Could you please give this PR another look ? |
3ae6f6b
to
f4ce0fc
Compare
f4ce0fc
to
7afb730
Compare
4d20c5c
to
fd75c4b
Compare
fd75c4b
to
bae9191
Compare
bae9191
to
3da4484
Compare
3da4484
to
467ca23
Compare
@gkalpak function errorHandlingConfig(config) {
if (isObject(config)) {
if (isDefined(config.objectMaxDepth)) {
objectMaxDepthInErrorMessage = isValidObjectMaxDepth(config.objectMaxDepth) ? config.objectMaxDepth : NaN;
}
} else {
return {
objectMaxDepth: objectMaxDepthInErrorMessage
};
}
} so it will work as a setter when config (object type ) is supplied. |
467ca23
to
6c8a548
Compare
6c8a548
to
d4b9de2
Compare
src/Angular.js
Outdated
* | ||
* @param {Object=} config - The configuration object. May only contain the options that need to be | ||
* updated. Supported keys: | ||
* - `objectMaxDepth` **{Number}** - The max depth for stringifying objects. Setting to a |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why is the dash
at the beginning of the line? Is it intentional?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I imagine it might have been cut-and-pasted from a diff?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What if you change line 148 to remove the indentation too?
d4b9de2
to
d19b038
Compare
@petebacondarwin |
d19b038
to
52de1d2
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We discussed this during the team meeting. I suggested two minor changes and then this willbe good to go.
src/Angular.js
Outdated
@@ -807,6 +855,7 @@ function arrayRemove(array, value) { | |||
* Can be any type, including primitives, `null`, and `undefined`. | |||
* @param {(Object|Array)=} destination Destination into which the source is copied. If | |||
* provided, must be of the same type as `source`. | |||
* @param {Number=} maxDepth All properties of the source will be copied until reaching the max depth. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We discussed this internally and we decided that it is better not to make this parameter public at this point.
So, please remove any mentions from the copy
docs.
src/Angular.js
Outdated
@@ -125,6 +128,50 @@ var VALIDITY_STATE_PROPERTY = 'validity'; | |||
|
|||
var hasOwnProperty = Object.prototype.hasOwnProperty; | |||
|
|||
var objectMaxDepthInErrorMessage = 5; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please change this to be an object with one property per config option. E.g.:
var minErrConfig = {
objectMaxDepth: 5
};
52de1d2
to
15ff3e6
Compare
@gkalpak
var objectMaxDepthInErrorMessage = 5; To be var minErrConfig = {
objectMaxDepth: 5
};
|
Regarding my earlier comment, it turns out that we don't need to change ng-closure-runner/minErr, because the changes in this PR are not applicable there. |
…in errors configurable Closes angular#15402 Closes angular#15433
What kind of change does this PR introduce? (Bug fix, feature, docs update, ...)
feature
What is the current behavior? (You can also link to an open issue here)
#15402
What is the new behavior (if this is a feature change)?
Does this PR introduce a breaking change?
Please check if the PR fulfills these requirements
Other information:
closes #15402