@@ -36,48 +36,26 @@ module.exports = internals.clone = function (obj, options = {}, _seen = null) {
36
36
}
37
37
}
38
38
39
+ // Built-in object types
40
+
39
41
const baseProto = Types . getInternalProto ( obj ) ;
40
- let newObj ;
41
-
42
- switch ( baseProto ) {
43
- case Types . buffer :
44
- return Buffer && Buffer . from ( obj ) ; // $lab:coverage:ignore$
45
-
46
- case Types . date :
47
- return new Date ( obj . getTime ( ) ) ;
48
-
49
- case Types . regex :
50
- return new RegExp ( obj ) ;
51
-
52
- case Types . array :
53
- newObj = [ ] ;
54
- break ;
55
-
56
- default :
57
- if ( options . prototype !== false ) { // Defaults to true
58
- const proto = Object . getPrototypeOf ( obj ) ;
59
- if ( proto &&
60
- proto . isImmutable ) {
61
-
62
- return obj ;
63
- }
64
-
65
- if ( internals . needsProtoHack . has ( baseProto ) ) {
66
- newObj = new proto . constructor ( ) ;
67
- if ( proto !== baseProto ) {
68
- Object . setPrototypeOf ( newObj , proto ) ;
69
- }
70
- }
71
- else {
72
- newObj = Object . create ( proto ) ;
73
- }
74
- }
75
- else if ( internals . needsProtoHack . has ( baseProto ) ) {
76
- newObj = new baseProto . constructor ( ) ;
77
- }
78
- else {
79
- newObj = { } ;
80
- }
42
+ if ( baseProto === Types . buffer ) {
43
+ return Buffer && Buffer . from ( obj ) ; // $lab:coverage:ignore$
44
+ }
45
+
46
+ if ( baseProto === Types . date ) {
47
+ return new Date ( obj . getTime ( ) ) ;
48
+ }
49
+
50
+ if ( baseProto === Types . regex ) {
51
+ return new RegExp ( obj ) ;
52
+ }
53
+
54
+ // Generic objects
55
+
56
+ const newObj = internals . base ( obj , baseProto , options ) ;
57
+ if ( newObj === obj ) {
58
+ return obj ;
81
59
}
82
60
83
61
if ( seen ) {
@@ -96,35 +74,38 @@ module.exports = internals.clone = function (obj, options = {}, _seen = null) {
96
74
}
97
75
98
76
const keys = Utils . keys ( obj , options ) ;
99
- for ( let i = 0 ; i < keys . length ; ++ i ) {
100
- const key = keys [ i ] ;
101
-
77
+ for ( const key of keys ) {
102
78
if ( baseProto === Types . array &&
103
79
key === 'length' ) {
104
80
81
+ newObj . length = obj . length ;
105
82
continue ;
106
83
}
107
84
108
85
const descriptor = Object . getOwnPropertyDescriptor ( obj , key ) ;
109
- if ( descriptor &&
110
- ( descriptor . get || descriptor . set ) ) {
86
+ if ( descriptor ) {
87
+ if ( descriptor . get ||
88
+ descriptor . set ) {
111
89
112
- Object . defineProperty ( newObj , key , descriptor ) ;
90
+ Object . defineProperty ( newObj , key , descriptor ) ;
91
+ }
92
+ else if ( descriptor . enumerable ) {
93
+ newObj [ key ] = clone ( obj [ key ] , options , seen ) ;
94
+ }
95
+ else {
96
+ Object . defineProperty ( newObj , key , { enumerable : false , writable : true , configurable : true , value : clone ( obj [ key ] , options , seen ) } ) ;
97
+ }
113
98
}
114
99
else {
115
100
Object . defineProperty ( newObj , key , {
116
- enumerable : descriptor ? descriptor . enumerable : true ,
101
+ enumerable : true ,
117
102
writable : true ,
118
103
configurable : true ,
119
104
value : clone ( obj [ key ] , options , seen )
120
105
} ) ;
121
106
}
122
107
}
123
108
124
- if ( baseProto === Types . array ) {
125
- newObj . length = obj . length ;
126
- }
127
-
128
109
return newObj ;
129
110
} ;
130
111
@@ -140,3 +121,37 @@ internals.cloneWithShallow = function (source, options) {
140
121
Utils . restore ( copy , source , storage ) ; // Shallow copy the stored items and restore
141
122
return copy ;
142
123
} ;
124
+
125
+
126
+ internals . base = function ( obj , baseProto , options ) {
127
+
128
+ if ( baseProto === Types . array ) {
129
+ return [ ] ;
130
+ }
131
+
132
+ if ( options . prototype === false ) { // Defaults to true
133
+ if ( internals . needsProtoHack . has ( baseProto ) ) {
134
+ return new baseProto . constructor ( ) ;
135
+ }
136
+
137
+ return { } ;
138
+ }
139
+
140
+ const proto = Object . getPrototypeOf ( obj ) ;
141
+ if ( proto &&
142
+ proto . isImmutable ) {
143
+
144
+ return obj ;
145
+ }
146
+
147
+ if ( internals . needsProtoHack . has ( baseProto ) ) {
148
+ const newObj = new proto . constructor ( ) ;
149
+ if ( proto !== baseProto ) {
150
+ Object . setPrototypeOf ( newObj , proto ) ;
151
+ }
152
+
153
+ return newObj ;
154
+ }
155
+
156
+ return Object . create ( proto ) ;
157
+ } ;
0 commit comments