From 96b3f8bc3f8428222ffa132c0c2a252d5a14b408 Mon Sep 17 00:00:00 2001 From: Damith Karunaratne Date: Fri, 14 May 2021 19:25:19 +0530 Subject: [PATCH 1/8] adding mergeChanges method to sakota --- src/__tests__/index.spec.ts | 97 +++++++++++++++++++++++++++++++++++++ src/index.ts | 53 ++++++++++++++++++++ 2 files changed, 150 insertions(+) diff --git a/src/__tests__/index.spec.ts b/src/__tests__/index.spec.ts index 08d55cc..6460570 100644 --- a/src/__tests__/index.spec.ts +++ b/src/__tests__/index.spec.ts @@ -623,4 +623,101 @@ describe('Sakota', () => { expect(console.warn).toHaveBeenCalledTimes(1); }); }); + + // Test for mergeChanges + // ----------------------- + + describe('mergeChanges', () => { + it('should merge given changes into the sakota model', () => { + const source = { + a: 123, + a1: 23, + b: { + x: 234, + y: 345, + }, + b1: { + x: 234, + y: 345, + }, + c: [{ a: 123 }], + d: [1, 2, 3], + e: [{ a: 123 }], + }; + + const target = { + a: 234, + a2: 23, + b: { + x: 234, + z: 234, + }, + b1: { + x: 234, + }, + c: [{ b: 123 }, { a: 234 }], + d: [1, 3], + e: [{ a: 234 }], + }; + + const wrapped: any = Sakota.create(source); + wrapped.a = 234; + delete wrapped.a1; + wrapped.a2 = 23; + wrapped.b.x = 234; + delete wrapped.b.y; + wrapped.c = [{ b: 123 }, { a: 234 }]; + wrapped.b.z = 234; + wrapped.d = [1, 3]; + wrapped.e[0].a = 234; + delete wrapped.b1.y; + + const wrapped1: any = Sakota.create(source); + wrapped1.__sakota__.mergeChanges(wrapped.__sakota__.getChanges()); + expect(wrapped1).toEqual(target as any); + }); + + it('should merge given changes into the existing sakota changes', () => { + const source = { + a: 123, + a1: 23, + b: { + x: 234, + y: 345, + }, + c: [{ a: 123 }], + d: [1, 2, 3], + e: [{ a: 123 }, { b: 123 }], + }; + + const target = { + a: 234, + a2: 23, + b: { + x: 234, + z: 234, + }, + c: [{ b: 123 }, { a: 234 }], + d: [1, 3], + e: [{ a: 234 }, { b: 345 }], + }; + + const wrapped: any = Sakota.create(source); + wrapped.a = 234; + delete wrapped.a1; + wrapped.a2 = 23; + wrapped.b.x = 234; + delete wrapped.b.y; + wrapped.c = [{ b: 123 }, { a: 234 }]; + + const wrapped1: any = Sakota.create(source); + wrapped1.b.z = 234; + wrapped1.d = [1, 3]; + wrapped1.e[0].a = 234; + wrapped1.e[1].b = 345; + + wrapped.__sakota__.mergeChanges(wrapped1.__sakota__.getChanges()); + expect(wrapped).toEqual(target as any); + }); + }); }); diff --git a/src/index.ts b/src/index.ts index fa67012..5137c71 100644 --- a/src/index.ts +++ b/src/index.ts @@ -314,6 +314,59 @@ export class Sakota implements ProxyHandler { return pattern ? this.filterChanges(changes, pattern) : changes; } + public mergeChanges(changes: Changes) { + const diff = this.diff || { $set: {}, $unset: {} }; + const kidChanges: { [prefix: string]: Changes } = {}; + if (changes.$set) { + Object.keys(changes.$set).forEach(key => { + const dotIndex = key.indexOf('.'); + if (dotIndex === -1) { + delete diff.$unset[key]; + delete this.kids[key]; + diff.$set[key] = changes.$set[key]; + } else { + const kkey = key.substring(0, dotIndex); + if (kidChanges.hasOwnProperty(kkey)) { + kidChanges[kkey].$set[key.substring(dotIndex + 1)] = changes.$set[key]; + } else { + kidChanges[kkey] = { + $set: { [key.substring(dotIndex + 1)]: changes.$set[key] }, + $unset: {}, + }; + } + } + }); + } + if (changes.$unset) { + Object.keys(changes.$unset).forEach(key => { + const dotIndex = key.indexOf('.'); + if (dotIndex === -1) { + delete diff.$set[key]; + delete this.kids[key]; + diff.$unset[key] = true; + } else { + const kkey = key.substring(0, dotIndex); + if (kidChanges.hasOwnProperty(kkey)) { + kidChanges[kkey].$unset[key.substring(dotIndex + 1)] = true; + } else { + kidChanges[kkey] = { + $set: {}, + $unset: { [key.substring(dotIndex + 1)]: true }, + }; + } + } + }); + } + + Object.keys(kidChanges).forEach(k => { + this.getKid(k, (this.target as any)[k]).__sakota__.mergeChanges(kidChanges[k]); + }); + + this.diff = diff; + this.changed = true; + this.changes = {}; + } + /** * Resets changes recorded in the proxy. Can be filtered by key name. */ From 724e44b062bb7b2b74fe1505d8609954c46256c9 Mon Sep 17 00:00:00 2001 From: Damith Karunaratne Date: Fri, 14 May 2021 19:47:11 +0530 Subject: [PATCH 2/8] using lodash isEqual package to replace deepEqual --- changelog.md | 5 +++++ package-lock.json | 22 +++++++++++++++++++++- package.json | 6 +++++- src/index.ts | 4 ++-- tsconfig.json | 1 + 5 files changed, 34 insertions(+), 4 deletions(-) diff --git a/changelog.md b/changelog.md index a5ff1e5..bc97b63 100644 --- a/changelog.md +++ b/changelog.md @@ -1,3 +1,8 @@ +# 2021-05-14 - v2.4.4 + + - added `mergeChanges` method to sakota + - replace `deepEqual` with `lodash.isequal` + # 2021-03-16 - v2.4.3 - improved change tracking. now if the value of a property is updated with the same value it will not track as a change. diff --git a/package-lock.json b/package-lock.json index 8be2536..2e44d20 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "@creately/sakota", - "version": "2.4.3", + "version": "2.4.4", "lockfileVersion": 1, "requires": true, "dependencies": { @@ -182,6 +182,21 @@ "integrity": "sha512-vw3VyFPa9mlba6NZPBZC3q2Zrnkgy5xuCVI43/tTLX6umdYrYvcFtQUKi2zH3PjFZQ9XCxNM/NMrM9uk8TPOzg==", "dev": true }, + "@types/lodash": { + "version": "4.14.169", + "resolved": "https://registry.npmjs.org/@types/lodash/-/lodash-4.14.169.tgz", + "integrity": "sha512-DvmZHoHTFJ8zhVYwCLWbQ7uAbYQEk52Ev2/ZiQ7Y7gQGeV9pjBqjnQpECMHfKS1rCYAhMI7LHVxwyZLZinJgdw==", + "dev": true + }, + "@types/lodash.isequal": { + "version": "4.5.5", + "resolved": "https://registry.npmjs.org/@types/lodash.isequal/-/lodash.isequal-4.5.5.tgz", + "integrity": "sha512-4IKbinG7MGP131wRfceK6W4E/Qt3qssEFLF30LnJbjYiSfHGGRU/Io8YxXrZX109ir+iDETC8hw8QsDijukUVg==", + "dev": true, + "requires": { + "@types/lodash": "*" + } + }, "@webassemblyjs/ast": { "version": "1.8.5", "resolved": "https://registry.npmjs.org/@webassemblyjs/ast/-/ast-1.8.5.tgz", @@ -3576,6 +3591,11 @@ "integrity": "sha512-UejweD1pDoXu+AD825lWwp4ZGtSwgnpZxb3JDViD7StjQz+Nb/6l093lx4OQ0foGWNRoc19mWy7BzL+UAK2iVg==", "dev": true }, + "lodash.isequal": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.isequal/-/lodash.isequal-4.5.0.tgz", + "integrity": "sha1-QVxEePK8wwEgwizhDtMib30+GOA=" + }, "log-symbols": { "version": "2.2.0", "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-2.2.0.tgz", diff --git a/package.json b/package.json index 511d36d..afeecae 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "@creately/sakota", - "version": "2.4.3", + "version": "2.4.4", "description": "Proxies js objects and records all changes made on an object without modifying the object.", "main": "dist/index.js", "typings": "dist/index.d.ts", @@ -15,6 +15,7 @@ "license": "MIT", "devDependencies": { "@types/jasmine": "^3.3.9", + "@types/lodash.isequal": "^4.5.5", "istanbul-instrumenter-loader": "^3.0.1", "jasmine-core": "^3.3.0", "karma": "^3.1.4", @@ -27,5 +28,8 @@ "ts-loader": "^5.3.3", "typescript": "^3.3.3333", "webpack": "^4.29.6" + }, + "dependencies": { + "lodash.isequal": "^4.5.0" } } diff --git a/src/index.ts b/src/index.ts index 5137c71..3f782d0 100644 --- a/src/index.ts +++ b/src/index.ts @@ -1,4 +1,4 @@ -import deepEqual from './deep-equal'; +import isEqual from 'lodash.isequal'; /** * The key used to get the handler. @@ -239,7 +239,7 @@ export class Sakota implements ProxyHandler { if (!this.diff) { this.diff = { $set: {}, $unset: {} }; } - if (key in obj && deepEqual(obj[key], val)) { + if (key in obj && isEqual(obj[key], val)) { if (this.diff.$unset[key] || this.diff.$set[key]) { delete this.diff.$unset[key]; delete this.diff.$set[key]; diff --git a/tsconfig.json b/tsconfig.json index 72e88ef..6a95c92 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -2,6 +2,7 @@ "compilerOptions": { "alwaysStrict": true, "declaration": true, + "esModuleInterop": true, "emitDecoratorMetadata": true, "experimentalDecorators": true, "lib": [ "dom", "es2015" ], From 2f885bd06f032775dd8a3be882baa30edeb416b9 Mon Sep 17 00:00:00 2001 From: Damith Karunaratne Date: Fri, 14 May 2021 23:05:55 +0530 Subject: [PATCH 3/8] removing files that are no longer needed --- src/deep-equal.ts | 26 -------------------------- 1 file changed, 26 deletions(-) delete mode 100644 src/deep-equal.ts diff --git a/src/deep-equal.ts b/src/deep-equal.ts deleted file mode 100644 index b972194..0000000 --- a/src/deep-equal.ts +++ /dev/null @@ -1,26 +0,0 @@ -// ref: https://stackoverflow.com/a/25456134 - -const deepEqual = (x: any, y: any) => { - if (x === y) { - return true; - } else if (typeof x == 'object' && x != null && (typeof y == 'object' && y != null)) { - if (Object.keys(x).length != Object.keys(y).length) { - return false; - } - - for (var prop in x) { - if (y.hasOwnProperty(prop)) { - if (!deepEqual(x[prop], y[prop])) { - return false; - } - } else { - return false; - } - } - - return true; - } - return false; -}; - -export default deepEqual; From b1b27f0b020a3ec374051ef9b071d80bef6bbf22 Mon Sep 17 00:00:00 2001 From: Damith Karunaratne Date: Mon, 17 May 2021 12:49:29 +0530 Subject: [PATCH 4/8] updating doc block for mergeChanges method --- src/index.ts | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/src/index.ts b/src/index.ts index 3f782d0..3633bc8 100644 --- a/src/index.ts +++ b/src/index.ts @@ -314,6 +314,12 @@ export class Sakota implements ProxyHandler { return pattern ? this.filterChanges(changes, pattern) : changes; } + /** + * This is an internal method to merge the changes from a different sakota object. + * If there are 2 skota objects, and one is modified and the same modification + * needs to be applied to the other object this method can be used. + * @param changes changes in Sakota format. + */ public mergeChanges(changes: Changes) { const diff = this.diff || { $set: {}, $unset: {} }; const kidChanges: { [prefix: string]: Changes } = {}; From 70d9f383c9ee3a94faae4329edba588ef6bb9785 Mon Sep 17 00:00:00 2001 From: Damith Karunaratne Date: Mon, 17 May 2021 15:58:46 +0530 Subject: [PATCH 5/8] corrected method argument types --- src/index.ts | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/src/index.ts b/src/index.ts index 3633bc8..8bc137d 100644 --- a/src/index.ts +++ b/src/index.ts @@ -320,23 +320,27 @@ export class Sakota implements ProxyHandler { * needs to be applied to the other object this method can be used. * @param changes changes in Sakota format. */ - public mergeChanges(changes: Changes) { + public mergeChanges(changes: Partial) { + if (Object.keys(changes).length === 0) { + return; + } const diff = this.diff || { $set: {}, $unset: {} }; const kidChanges: { [prefix: string]: Changes } = {}; if (changes.$set) { + const $set = changes.$set; Object.keys(changes.$set).forEach(key => { const dotIndex = key.indexOf('.'); if (dotIndex === -1) { delete diff.$unset[key]; delete this.kids[key]; - diff.$set[key] = changes.$set[key]; + diff.$set[key] = $set[key]; } else { const kkey = key.substring(0, dotIndex); if (kidChanges.hasOwnProperty(kkey)) { - kidChanges[kkey].$set[key.substring(dotIndex + 1)] = changes.$set[key]; + kidChanges[kkey].$set[key.substring(dotIndex + 1)] = $set[key]; } else { kidChanges[kkey] = { - $set: { [key.substring(dotIndex + 1)]: changes.$set[key] }, + $set: { [key.substring(dotIndex + 1)]: $set[key] }, $unset: {}, }; } From 8ef6411326c05de3c056ce3401359c5fc6a65b9e Mon Sep 17 00:00:00 2001 From: Damith Karunaratne Date: Tue, 25 May 2021 18:52:35 +0530 Subject: [PATCH 6/8] adding unwrap method to Sakota --- changelog.md | 2 +- src/__tests__/index.spec.ts | 91 +++++++++++++++++++++++++++++++++++++ src/index.ts | 22 +++++++++ 3 files changed, 114 insertions(+), 1 deletion(-) diff --git a/changelog.md b/changelog.md index bc97b63..3b434ee 100644 --- a/changelog.md +++ b/changelog.md @@ -1,6 +1,6 @@ # 2021-05-14 - v2.4.4 - - added `mergeChanges` method to sakota + - added `mergeChanges` and `unwrap` method to sakota - replace `deepEqual` with `lodash.isequal` # 2021-03-16 - v2.4.3 diff --git a/src/__tests__/index.spec.ts b/src/__tests__/index.spec.ts index 6460570..ee4d6ea 100644 --- a/src/__tests__/index.spec.ts +++ b/src/__tests__/index.spec.ts @@ -720,4 +720,95 @@ describe('Sakota', () => { expect(wrapped).toEqual(target as any); }); }); + + // Test for unwrap + // ----------------------- + describe('unwrap', () => { + it('should create a copy of the object removing sakota', () => { + const obj: any = { + a: 123, + b: { + c: 234, + }, + }; + const wrapped = Sakota.create(freeze(obj)); + wrapped.a = 345; + wrapped.a1 = 234; + wrapped.b.c = 2345; + + const expected = { + a: 345, + a1: 234, + b: { + c: 2345, + }, + }; + + const unwrapped = wrapped.__sakota__.unwrap(); + expect(unwrapped).toEqual(expected); + expect(unwrapped === obj).toBeFalsy(); + expect(unwrapped.__sakota__).toBeUndefined(); + expect(unwrapped.b.__sakota__).toBeUndefined(); + }); + + it('should apply the changes to the target object if unwrapped in place', () => { + const obj: any = { + a: 123, + b: { + c: 234, + }, + }; + const wrapped = Sakota.create(obj); + wrapped.a = 345; + wrapped.a1 = 234; + wrapped.b.c = 2345; + + const expected = { + a: 345, + a1: 234, + b: { + c: 2345, + }, + }; + + const unwrapped = wrapped.__sakota__.unwrap(true); + expect(unwrapped === obj).toBeTruthy(); + expect(obj).toEqual(expected); + expect(obj.__sakota__).toBeUndefined(); + expect(obj.b.__sakota__).toBeUndefined(); + }); + + it('should remove Sakota wrapper around array props', () => { + const obj: any = { + a: [{ b: 234 }], + }; + const wrapped = Sakota.create(freeze(obj)); + wrapped.a[0].b = 345; + const expected = { + a: [{ b: 345 }], + }; + + const unwrapped = wrapped.__sakota__.unwrap(); + expect(unwrapped).toEqual(expected); + expect(unwrapped === obj).toBeFalsy(); + expect(unwrapped.__sakota__).toBeUndefined(); + expect(unwrapped.a.__sakota__).toBeUndefined(); + expect(unwrapped.a[0].__sakota__).toBeUndefined(); + }); + + it('should return the same object as target', () => { + const obj: any = { + a: [{ b: 234 }], + }; + const wrapped = Sakota.create(obj); + delete obj.a[0].b; + + const unwrapped = wrapped.__sakota__.unwrap(true); + expect(unwrapped === obj).toBeTruthy(); + expect(unwrapped).toEqual({ a: [{}] }); + expect(unwrapped.__sakota__).toBeUndefined(); + expect(unwrapped.a.__sakota__).toBeUndefined(); + expect(unwrapped.a[0].__sakota__).toBeUndefined(); + }); + }); }); diff --git a/src/index.ts b/src/index.ts index 8bc137d..499d69a 100644 --- a/src/index.ts +++ b/src/index.ts @@ -394,6 +394,28 @@ export class Sakota implements ProxyHandler { this.onChange(); } + /** + * this method removes Sakota wrapper for the target object + * @param inplace if true modifies the target object otherwise returns a copy of the target + * @returns Sakota wrapper removed object + */ + public unwrap(inplace: boolean = false) { + const $set = this.diff ? Object.assign({}, this.diff.$set) : {}; + const $unset = this.diff ? Object.keys(this.diff.$unset) : []; + Object.keys(this.kids).forEach(k => { + $set[k] = this.kids[k].__sakota__.unwrap(inplace); + }); + let val: any; + if (Array.isArray(this.target)) { + val = inplace ? this.target : this.target.slice(); + Object.keys($set).forEach(k => (val[k] = $set[k])); + } else { + val = inplace ? Object.assign(this.target, $set) : Object.assign({}, this.target, $set); + } + $unset.forEach(k => delete val[k]); + return val; + } + // Private Methods // --------------- From 16b274eb82b8bec6392bba9d892418a2f07ac15f Mon Sep 17 00:00:00 2001 From: Damith Karunaratne Date: Sun, 6 Jun 2021 23:51:14 +0530 Subject: [PATCH 7/8] corrected mergeChanges method --- package-lock.json | 14 +++++++++++ package.json | 4 ++- src/__tests__/index.spec.ts | 31 +++++++++++++++++++++++ src/index.ts | 49 ++++++++++++++++++++++++++++++++++++- 4 files changed, 96 insertions(+), 2 deletions(-) diff --git a/package-lock.json b/package-lock.json index 2e44d20..71eb9e0 100644 --- a/package-lock.json +++ b/package-lock.json @@ -197,6 +197,15 @@ "@types/lodash": "*" } }, + "@types/lodash.set": { + "version": "4.3.6", + "resolved": "https://registry.npmjs.org/@types/lodash.set/-/lodash.set-4.3.6.tgz", + "integrity": "sha512-ZeGDDlnRYTvS31Laij0RsSaguIUSBTYIlJFKL3vm3T2OAZAQj2YpSvVWJc0WiG4jqg9fGX6PAPGvDqBcHfSgFg==", + "dev": true, + "requires": { + "@types/lodash": "*" + } + }, "@webassemblyjs/ast": { "version": "1.8.5", "resolved": "https://registry.npmjs.org/@webassemblyjs/ast/-/ast-1.8.5.tgz", @@ -3596,6 +3605,11 @@ "resolved": "https://registry.npmjs.org/lodash.isequal/-/lodash.isequal-4.5.0.tgz", "integrity": "sha1-QVxEePK8wwEgwizhDtMib30+GOA=" }, + "lodash.set": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/lodash.set/-/lodash.set-4.3.2.tgz", + "integrity": "sha1-2HV7HagH3eJIFrDWqEvqGnYjCyM=" + }, "log-symbols": { "version": "2.2.0", "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-2.2.0.tgz", diff --git a/package.json b/package.json index afeecae..263f2ba 100644 --- a/package.json +++ b/package.json @@ -16,6 +16,7 @@ "devDependencies": { "@types/jasmine": "^3.3.9", "@types/lodash.isequal": "^4.5.5", + "@types/lodash.set": "^4.3.6", "istanbul-instrumenter-loader": "^3.0.1", "jasmine-core": "^3.3.0", "karma": "^3.1.4", @@ -30,6 +31,7 @@ "webpack": "^4.29.6" }, "dependencies": { - "lodash.isequal": "^4.5.0" + "lodash.isequal": "^4.5.0", + "lodash.set": "^4.3.2" } } diff --git a/src/__tests__/index.spec.ts b/src/__tests__/index.spec.ts index ee4d6ea..bd111f5 100644 --- a/src/__tests__/index.spec.ts +++ b/src/__tests__/index.spec.ts @@ -811,4 +811,35 @@ describe('Sakota', () => { expect(unwrapped.a[0].__sakota__).toBeUndefined(); }); }); + + describe('mergeChanges + unwrap', () => { + it('should handle multiple changes properly', () => { + const source = { + a: 123, + }; + const target = { + a: 234, + b: { + c: 345, + }, + c: { + d: 456, + }, + }; + const sakotaWrapped: any = Sakota.create(source); + sakotaWrapped.a = 234; + sakotaWrapped.b = { + c: 234, + d: 345, + }; + const sakotaWrapped1 = Sakota.create(sakotaWrapped.__sakota__.unwrap()); + sakotaWrapped1.b.c = 345; + delete sakotaWrapped1.b.d; + sakotaWrapped1.c = { d: 234 }; + sakotaWrapped1.c.d = 456; + sakotaWrapped.__sakota__.mergeChanges(sakotaWrapped1.__sakota__.getChanges()); + expect(sakotaWrapped.__sakota__.unwrap()).toEqual(sakotaWrapped); + expect(sakotaWrapped).toEqual(target); + }); + }); }); diff --git a/src/index.ts b/src/index.ts index 499d69a..4a47b8a 100644 --- a/src/index.ts +++ b/src/index.ts @@ -1,4 +1,5 @@ import isEqual from 'lodash.isequal'; +import _set from 'lodash.set'; /** * The key used to get the handler. @@ -369,7 +370,24 @@ export class Sakota implements ProxyHandler { } Object.keys(kidChanges).forEach(k => { - this.getKid(k, (this.target as any)[k]).__sakota__.mergeChanges(kidChanges[k]); + if (diff.$set.hasOwnProperty(k)) { + /* istanbul ignore if */ + if (typeof diff.$set[k] !== 'object') { + throw new Error('Invalid modifier'); // this scenario is not expected. + } + this.applyModifier(diff.$set[k], kidChanges[k]); + } else if (this.target.hasOwnProperty(k)) { + this.getKid(k, (this.target as any)[k]).__sakota__.mergeChanges(kidChanges[k]); + } else { + console.warn('unexpected modifier', { path: k, modifier: changes }); + const skeys = Object.keys(kidChanges[k].$set); + const ukeys = Object.keys(kidChanges[k].$set); + if (skeys.length === 0 || ukeys.length > 0 || skeys.some(k => k.includes('.'))) { + throw new Error('Invalid modifier'); // this scenario is not expected. + } else { + diff.$set[k] = kidChanges[k].$set; + } + } }); this.diff = diff; @@ -377,6 +395,35 @@ export class Sakota implements ProxyHandler { this.changes = {}; } + /** + * applying Sakota diff to an object inplace. + * this is similar to @creately/mungo::modify method. + * @param obj + * @param modifier + */ + private applyModifier(obj: any, modifier: Changes) { + Object.keys(modifier.$set).forEach(k => { + _set(obj, k.split('.'), modifier.$set[k]); + }); + Object.keys(modifier.$unset).forEach(k => { + if (k.includes('.')) { + const path = k.split('.'); + k = path.pop() as string; + delete this._get(obj, path)[k]; + } else { + delete obj[k]; + } + }); + } + + private _get(obj: any, path: string[]): any { + if (path.length === 0) { + return obj; + } + const [prop, ...remainingPath] = path; + return this._get(obj[prop], remainingPath); + } + /** * Resets changes recorded in the proxy. Can be filtered by key name. */ From b96b6c33711c8f82eb6d16c381c2fc6902ac391b Mon Sep 17 00:00:00 2001 From: Damith Karunaratne Date: Mon, 7 Jun 2021 17:58:35 +0530 Subject: [PATCH 8/8] change access modifiers for hasSakota method --- changelog.md | 1 + src/__tests__/index.spec.ts | 14 ++++---------- src/index.ts | 8 ++++---- 3 files changed, 9 insertions(+), 14 deletions(-) diff --git a/changelog.md b/changelog.md index 3b434ee..28d8bf0 100644 --- a/changelog.md +++ b/changelog.md @@ -2,6 +2,7 @@ - added `mergeChanges` and `unwrap` method to sakota - replace `deepEqual` with `lodash.isequal` + - added `hasSakota` a public static method # 2021-03-16 - v2.4.3 diff --git a/src/__tests__/index.spec.ts b/src/__tests__/index.spec.ts index bd111f5..e6a1449 100644 --- a/src/__tests__/index.spec.ts +++ b/src/__tests__/index.spec.ts @@ -747,8 +747,7 @@ describe('Sakota', () => { const unwrapped = wrapped.__sakota__.unwrap(); expect(unwrapped).toEqual(expected); expect(unwrapped === obj).toBeFalsy(); - expect(unwrapped.__sakota__).toBeUndefined(); - expect(unwrapped.b.__sakota__).toBeUndefined(); + expect(Sakota.hasSakota(unwrapped)).toBeFalsy(); }); it('should apply the changes to the target object if unwrapped in place', () => { @@ -774,8 +773,7 @@ describe('Sakota', () => { const unwrapped = wrapped.__sakota__.unwrap(true); expect(unwrapped === obj).toBeTruthy(); expect(obj).toEqual(expected); - expect(obj.__sakota__).toBeUndefined(); - expect(obj.b.__sakota__).toBeUndefined(); + expect(Sakota.hasSakota(unwrapped)).toBeFalsy(); }); it('should remove Sakota wrapper around array props', () => { @@ -791,9 +789,7 @@ describe('Sakota', () => { const unwrapped = wrapped.__sakota__.unwrap(); expect(unwrapped).toEqual(expected); expect(unwrapped === obj).toBeFalsy(); - expect(unwrapped.__sakota__).toBeUndefined(); - expect(unwrapped.a.__sakota__).toBeUndefined(); - expect(unwrapped.a[0].__sakota__).toBeUndefined(); + expect(Sakota.hasSakota(unwrapped)).toBeFalsy(); }); it('should return the same object as target', () => { @@ -806,9 +802,7 @@ describe('Sakota', () => { const unwrapped = wrapped.__sakota__.unwrap(true); expect(unwrapped === obj).toBeTruthy(); expect(unwrapped).toEqual({ a: [{}] }); - expect(unwrapped.__sakota__).toBeUndefined(); - expect(unwrapped.a.__sakota__).toBeUndefined(); - expect(unwrapped.a[0].__sakota__).toBeUndefined(); + expect(Sakota.hasSakota(unwrapped)).toBeFalsy(); }); }); diff --git a/src/index.ts b/src/index.ts index 4a47b8a..30ccb58 100644 --- a/src/index.ts +++ b/src/index.ts @@ -226,7 +226,7 @@ export class Sakota implements ProxyHandler { */ public set(obj: any, key: KeyType, val: any): boolean { if (!Sakota.config.prodmode) { - if (this._hasSakota(val)) { + if (Sakota.hasSakota(val)) { console.warn('Sakota: value is also wrapped by Sakota!', { obj: obj, key, val }); } } @@ -645,7 +645,7 @@ export class Sakota implements ProxyHandler { /** * Checks whether the value or it's children is proxied with Sakota. */ - private _hasSakota(value: unknown): boolean { + public static hasSakota(value: unknown): boolean { if (typeof value !== 'object') { return false; } @@ -657,7 +657,7 @@ export class Sakota implements ProxyHandler { } if (Array.isArray(value)) { for (const child of value) { - if (this._hasSakota(child)) { + if (Sakota.hasSakota(child)) { return true; } } @@ -665,7 +665,7 @@ export class Sakota implements ProxyHandler { } for (const key in value) { const child = (value as any)[key]; - if (this._hasSakota(child)) { + if (Sakota.hasSakota(child)) { return true; } }