From 3535bd2511c9d8773adf16b6555704c577ad2a36 Mon Sep 17 00:00:00 2001 From: Gil Pedersen Date: Thu, 14 Nov 2024 16:15:29 +0100 Subject: [PATCH] Fix cloning Util.inherit() subclassed errors. Closes #400 --- lib/clone.js | 8 +++++--- test/clone.js | 29 +++++++++++++++++++++++++++++ 2 files changed, 34 insertions(+), 3 deletions(-) diff --git a/lib/clone.js b/lib/clone.js index 2e5aebd..b2da579 100755 --- a/lib/clone.js +++ b/lib/clone.js @@ -166,10 +166,12 @@ internals.base = function (obj, baseProto, options) { return newObj; } - else if (baseProto === Types.error) { - const err = structuredClone(obj); // Needed to copy internal stack state + else if (baseProto === Types.error && + (proto === baseProto || Error.isPrototypeOf(proto.constructor))) { // Don't match Util.inherit() subclassed errors + + const err = structuredClone(obj); // Needed to copy internal stack state if (Object.getPrototypeOf(err) !== proto) { - Object.setPrototypeOf(err, proto); // Fix prototype + Object.setPrototypeOf(err, proto); // Fix prototype } return err; diff --git a/test/clone.js b/test/clone.js index a6823e3..2752e4e 100755 --- a/test/clone.js +++ b/test/clone.js @@ -731,6 +731,35 @@ describe('clone()', () => { expect(b.stack).to.equal(a.stack); }); + it('clones Error with function property', () => { + + const a = new Error('hello'); + a.fun = new Function(); + + const b = Hoek.clone(a); + + expect(b).to.equal(a); + expect(b.stack).to.equal(a.stack); + }); + + it('clones legacy extended Error with function property', () => { + + const CustomError = function () { + + Error.call(this); + }; + + Object.setPrototypeOf(CustomError.prototype, Error.prototype); + + const a = new CustomError('hello'); + a.fun = new Function(); + + const b = Hoek.clone(a); + + expect(b).to.equal(a); + expect(b.stack).to.equal(a.stack); + }); + it('cloned Error handles late stack update', () => { const a = new Error('bad');