10
10
splice,
11
11
push,
12
12
toString,
13
+ objectMaxDepthInErrorMessage,
14
+ errorHandlingConfig,
15
+ isValidObjectMaxDepth,
13
16
ngMinErr,
14
17
angularModule,
15
18
uid,
@@ -125,6 +128,49 @@ var VALIDITY_STATE_PROPERTY = 'validity';
125
128
126
129
var hasOwnProperty = Object . prototype . hasOwnProperty ;
127
130
131
+ var objectMaxDepthInErrorMessage = 5 ;
132
+
133
+ /**
134
+ * @ngdoc function
135
+ * @name angular.errorHandlingConfig
136
+ * @module ng
137
+ * @kind function
138
+ *
139
+ * @description
140
+ * Configure several aspects of error handling in AngularJS if used as a setter or return the
141
+ * current configuration if used as a getter. The following options are supported:
142
+ *
143
+ * - **objectMaxDepth**: The maximum depth to which objects are traversed when stringified for error messages.
144
+ *
145
+ * Omitted or undefined options will leave the corresponding configuration values unchanged.
146
+ *
147
+ * @param {Object= } config - The configuration object. May only contain the options that need to be
148
+ * updated. Supported keys:
149
+ * - `objectMaxDepth` **{Number}** - The max depth for stringifying objects. Setting to a
150
+ * non-positive or non-numeric value, removes the max depth limit.
151
+ * Default: 5
152
+ */
153
+ function errorHandlingConfig ( config ) {
154
+ if ( isObject ( config ) ) {
155
+ if ( isDefined ( config . objectMaxDepth ) ) {
156
+ objectMaxDepthInErrorMessage = isValidObjectMaxDepth ( config . objectMaxDepth ) ? config . objectMaxDepth : NaN ;
157
+ }
158
+ } else {
159
+ return {
160
+ objectMaxDepth : objectMaxDepthInErrorMessage
161
+ } ;
162
+ }
163
+ }
164
+
165
+ /**
166
+ * @private
167
+ * @param {Number } maxDepth
168
+ * @return {boolean }
169
+ */
170
+ function isValidObjectMaxDepth ( maxDepth ) {
171
+ return isNumber ( maxDepth ) && maxDepth > 0 ;
172
+ }
173
+
128
174
/**
129
175
* @ngdoc function
130
176
* @name angular.lowercase
@@ -796,6 +842,7 @@ function arrayRemove(array, value) {
796
842
* are deleted and then all elements/properties from the source are copied to it.
797
843
* * If `source` is not an object or array (inc. `null` and `undefined`), `source` is returned.
798
844
* * If `source` is identical to `destination` an exception will be thrown.
845
+ * * If `maxDepth` is supplied, all properties of the source will be copied until reaching the max depth.
799
846
*
800
847
* <br />
801
848
* <div class="alert alert-warning">
@@ -807,6 +854,7 @@ function arrayRemove(array, value) {
807
854
* Can be any type, including primitives, `null`, and `undefined`.
808
855
* @param {(Object|Array)= } destination Destination into which the source is copied. If
809
856
* provided, must be of the same type as `source`.
857
+ * @param {Number= } maxDepth All properties of the source will be copied until reaching the max depth.
810
858
* @returns {* } The copy or updated `destination`, if `destination` was specified.
811
859
*
812
860
* @example
@@ -847,9 +895,10 @@ function arrayRemove(array, value) {
847
895
</file>
848
896
</example>
849
897
*/
850
- function copy ( source , destination ) {
898
+ function copy ( source , destination , maxDepth ) {
851
899
var stackSource = [ ] ;
852
900
var stackDest = [ ] ;
901
+ maxDepth = isValidObjectMaxDepth ( maxDepth ) ? maxDepth : NaN ;
853
902
854
903
if ( destination ) {
855
904
if ( isTypedArray ( destination ) || isArrayBuffer ( destination ) ) {
@@ -872,43 +921,47 @@ function copy(source, destination) {
872
921
873
922
stackSource . push ( source ) ;
874
923
stackDest . push ( destination ) ;
875
- return copyRecurse ( source , destination ) ;
924
+ return copyRecurse ( source , destination , maxDepth ) ;
876
925
}
877
926
878
- return copyElement ( source ) ;
927
+ return copyElement ( source , maxDepth ) ;
879
928
880
- function copyRecurse ( source , destination ) {
929
+ function copyRecurse ( source , destination , maxDepth ) {
930
+ maxDepth -- ;
931
+ if ( maxDepth < 0 ) {
932
+ return '...' ;
933
+ }
881
934
var h = destination . $$hashKey ;
882
935
var key ;
883
936
if ( isArray ( source ) ) {
884
937
for ( var i = 0 , ii = source . length ; i < ii ; i ++ ) {
885
- destination . push ( copyElement ( source [ i ] ) ) ;
938
+ destination . push ( copyElement ( source [ i ] , maxDepth ) ) ;
886
939
}
887
940
} else if ( isBlankObject ( source ) ) {
888
941
// createMap() fast path --- Safe to avoid hasOwnProperty check because prototype chain is empty
889
942
for ( key in source ) {
890
- destination [ key ] = copyElement ( source [ key ] ) ;
943
+ destination [ key ] = copyElement ( source [ key ] , maxDepth ) ;
891
944
}
892
945
} else if ( source && typeof source . hasOwnProperty === 'function' ) {
893
946
// Slow path, which must rely on hasOwnProperty
894
947
for ( key in source ) {
895
948
if ( source . hasOwnProperty ( key ) ) {
896
- destination [ key ] = copyElement ( source [ key ] ) ;
949
+ destination [ key ] = copyElement ( source [ key ] , maxDepth ) ;
897
950
}
898
951
}
899
952
} else {
900
953
// Slowest path --- hasOwnProperty can't be called as a method
901
954
for ( key in source ) {
902
955
if ( hasOwnProperty . call ( source , key ) ) {
903
- destination [ key ] = copyElement ( source [ key ] ) ;
956
+ destination [ key ] = copyElement ( source [ key ] , maxDepth ) ;
904
957
}
905
958
}
906
959
}
907
960
setHashKey ( destination , h ) ;
908
961
return destination ;
909
962
}
910
963
911
- function copyElement ( source ) {
964
+ function copyElement ( source , maxDepth ) {
912
965
// Simple values
913
966
if ( ! isObject ( source ) ) {
914
967
return source ;
@@ -937,7 +990,7 @@ function copy(source, destination) {
937
990
stackDest . push ( destination ) ;
938
991
939
992
return needsRecurse
940
- ? copyRecurse ( source , destination )
993
+ ? copyRecurse ( source , destination , maxDepth )
941
994
: destination ;
942
995
}
943
996
0 commit comments