Skip to content
This repository was archived by the owner on Apr 12, 2024. It is now read-only.

feat(minErr): set max depth for angular error JSON stringify #15433

Closed
wants to merge 1 commit into from

Conversation

m-amr
Copy link
Contributor

@m-amr m-amr commented Nov 24, 2016

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

Copy link
Member

@gkalpak gkalpak left a 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:

  1. Tests.
  2. 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);
Copy link
Member

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;
Copy link
Member

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) {
Copy link
Member

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;
}
Copy link
Member

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;
Copy link
Member

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.

Copy link
Contributor

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;
Copy link
Member

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).

Copy link
Contributor Author

@m-amr m-amr Jan 21, 2017

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);
  1. Refactor exampleMinErr to be
throw exampleMinErr({code:'one', message: 'This {0} is {1}', messageArgs:[foo, boo]});

and we can keep both styles working.

  1. 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 ?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@petebacondarwin
Copy link
Contributor

Wouldn't it be simpler and more performant just to provide a specialized replacer function to the JSON.stringify method, which tracked depth? Or is that not possible?

@gkalpak
Copy link
Member

gkalpak commented Nov 25, 2016

I couldn't think of a way to track depth from jsonReplacer (in a reasonable way) 😞

@petebacondarwin
Copy link
Contributor

Yeah, fair enough.
This project https://github.com/Canop/JSON.prune ended up having to rewrite JSON.stringify completely!

@m-amr m-amr force-pushed the stringify_json_with_depth branch 9 times, most recently from fa1823e to d53eca0 Compare November 27, 2016 16:03
@m-amr m-amr force-pushed the stringify_json_with_depth branch from d53eca0 to 3ae6f6b Compare January 21, 2017 16:03
m-amr added a commit to m-amr/angular.js that referenced this pull request Jan 21, 2017
@m-amr
Copy link
Contributor Author

m-amr commented Jan 21, 2017

@gkalpak
I have added some unit tests and we can configure the max depth by passing it in the placeholder

var exampleMinErr = minErr('example');
throw exampleMinErr('one', 'This {0,maxDepth=2}', foo);

Could you please give this PR another look ?

@m-amr m-amr force-pushed the stringify_json_with_depth branch from 3ae6f6b to f4ce0fc Compare January 22, 2017 08:20
m-amr added a commit to m-amr/angular.js that referenced this pull request Jan 22, 2017
@m-amr m-amr force-pushed the stringify_json_with_depth branch from f4ce0fc to 7afb730 Compare January 22, 2017 13:53
m-amr added a commit to m-amr/angular.js that referenced this pull request Jan 22, 2017
@m-amr m-amr force-pushed the stringify_json_with_depth branch from 4d20c5c to fd75c4b Compare January 31, 2017 20:17
m-amr added a commit to m-amr/angular.js that referenced this pull request Jan 31, 2017
@m-amr m-amr force-pushed the stringify_json_with_depth branch from fd75c4b to bae9191 Compare January 31, 2017 21:25
m-amr added a commit to m-amr/angular.js that referenced this pull request Jan 31, 2017
@m-amr m-amr force-pushed the stringify_json_with_depth branch from bae9191 to 3da4484 Compare January 31, 2017 21:31
m-amr added a commit to m-amr/angular.js that referenced this pull request Jan 31, 2017
@m-amr m-amr force-pushed the stringify_json_with_depth branch from 3da4484 to 467ca23 Compare January 31, 2017 21:37
m-amr added a commit to m-amr/angular.js that referenced this pull request Jan 31, 2017
@m-amr
Copy link
Contributor Author

m-amr commented Jan 31, 2017

@gkalpak
I prefer your suggestion about using errorHandlingConfig as a getter and a setter,
I have changed the implementation to

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.
please tell me if there anything missing.

@m-amr m-amr force-pushed the stringify_json_with_depth branch from 467ca23 to 6c8a548 Compare February 1, 2017 00:33
m-amr added a commit to m-amr/angular.js that referenced this pull request Feb 1, 2017
@m-amr m-amr force-pushed the stringify_json_with_depth branch from 6c8a548 to d4b9de2 Compare February 1, 2017 00:52
m-amr added a commit to m-amr/angular.js that referenced this pull request Feb 1, 2017
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
Copy link
Member

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?

Copy link
Contributor

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?

Copy link
Contributor Author

@m-amr m-amr Feb 2, 2017

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Because when the dash is at the beginning of the line it will be converted to dot

screen shot 2017-02-03 at 12 32 39 am copy

which will be consistent with angular.bootsrtrap config docs

screen shot 2017-02-03 at 12 40 53 am copy

But when it is not at the beginning

screen shot 2017-02-03 at 12 37 37 am copy

Do you want to change it ?

Copy link
Contributor

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?

@m-amr m-amr force-pushed the stringify_json_with_depth branch from d4b9de2 to d19b038 Compare February 2, 2017 23:10
m-amr added a commit to m-amr/angular.js that referenced this pull request Feb 2, 2017
@m-amr
Copy link
Contributor Author

m-amr commented Feb 2, 2017

@petebacondarwin
@gkalpak
I have change it, it is now identical to the docs of 'angular.bootstrap' config in the identation
please tell me if it needs to change ?

@m-amr m-amr force-pushed the stringify_json_with_depth branch from d19b038 to 52de1d2 Compare February 3, 2017 00:12
m-amr added a commit to m-amr/angular.js that referenced this pull request Feb 3, 2017
Copy link
Member

@gkalpak gkalpak left a 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.
Copy link
Member

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;
Copy link
Member

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
};

@m-amr m-amr force-pushed the stringify_json_with_depth branch from 52de1d2 to 15ff3e6 Compare February 7, 2017 17:15
@m-amr
Copy link
Contributor Author

m-amr commented Feb 7, 2017

@gkalpak
I did all the minor changes that you had requested.

  1. Changed
var objectMaxDepthInErrorMessage = 5;

To be

var minErrConfig = {
  objectMaxDepth: 5
};
  1. Removed maxDepth from copy docs

@gkalpak
Copy link
Member

gkalpak commented Feb 8, 2017

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.

@gkalpak gkalpak closed this in a0641ea Feb 8, 2017
gkalpak pushed a commit that referenced this pull request Feb 8, 2017
ellimist pushed a commit to ellimist/angular.js that referenced this pull request Mar 15, 2017
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Feature Request: set max depth for angular error JSON stringify
4 participants