Skip to content
New issue

Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? # to your account

feat: Add two-way binding of Parse object properties with Parse.Object.bind.key = value #1484

Open
wants to merge 60 commits into
base: alpha
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
60 commits
Select commit Hold shift + click to select a range
dbf8202
Update ParseObject.js
dblythy May 24, 2022
f03352d
Update ParseObject.js
dblythy May 24, 2022
70fcefd
feat: allow dot notation for Parse Objects with `Parse.dotNotation = …
dblythy May 24, 2022
e944f91
Update ParseObjectTest.js
dblythy May 24, 2022
5044a30
Update ParseObject.js
dblythy May 24, 2022
75363d5
Update ParseObjectTest.js
dblythy May 24, 2022
302018c
Update ParseObject.js
dblythy May 24, 2022
31fbfcc
add user test
dblythy May 25, 2022
2c5ec36
Update ParseUserTest.js
dblythy May 25, 2022
953b637
Update ParseObjectTest.js
dblythy May 26, 2022
493cd09
Update ParseObject.js
dblythy May 26, 2022
ade003a
Update ParseObjectTest.js
dblythy May 26, 2022
007dc28
Merge branch 'alpha' into dot-notation
mtrezza May 27, 2022
a5751a1
return receiver
dblythy Jun 2, 2022
df73705
Merge branch 'alpha' into dot-notation
dblythy Jun 2, 2022
6aa25ec
Update ParseUserTest.js
dblythy Jun 2, 2022
d3857e9
add attribute check
dblythy Jun 3, 2022
b73c2b8
Update ParseObject.js
dblythy Jun 3, 2022
0cbf9d8
Merge branch 'alpha' into dot-notation
mtrezza Jun 30, 2022
76b3141
Merge branch 'alpha' into dot-notation
dblythy Jul 19, 2022
284b43a
Merge branch 'alpha' into dot-notation
dblythy Aug 28, 2022
d7bf41c
add delete trap
dblythy Aug 29, 2022
f32efeb
Update ParseObject.js
dblythy Sep 9, 2022
441218c
Update ParseObject.js
dblythy Sep 9, 2022
9fc5852
Merge branch 'alpha' into dot-notation
mtrezza Sep 9, 2022
1dd4943
add deep
dblythy Oct 6, 2022
bf5ed49
add dirtyKeys
dblythy Oct 7, 2022
505cf38
Update ParseObject.js
dblythy Oct 7, 2022
91735e8
Update ParseObject.js
dblythy Oct 10, 2022
86cac78
Update ParseObject.js
dblythy Oct 12, 2022
22587a0
Update ParseObject.js
dblythy Oct 13, 2022
5e6328c
Update ParseObject.js
dblythy Nov 4, 2022
878c177
Update ParseObject.js
dblythy Nov 4, 2022
5197e72
Update ParseObject.js
dblythy Nov 13, 2022
c4af730
Merge branch 'alpha' into dot-notation
dblythy Mar 3, 2023
44bcf34
wip
dblythy Mar 3, 2023
461f26c
wip
dblythy Mar 3, 2023
9b3c3c9
bind
dblythy Mar 3, 2023
2be3fef
tests
dblythy Mar 3, 2023
ec61f69
changes
dblythy Mar 5, 2023
2d24479
changes
dblythy Mar 5, 2023
e20f550
change
dblythy Mar 5, 2023
d38aee8
Merge branch 'alpha' into dot-notation
dblythy Mar 5, 2023
6b1716a
refactor to bind
dblythy Mar 6, 2023
d8da6e1
refactor to `bind`
dblythy Mar 6, 2023
dfc7b6f
refactor tests
dblythy Mar 6, 2023
65e2b0d
integration tests
dblythy Mar 6, 2023
ab99429
Update ParseObjectTest.js
dblythy Mar 6, 2023
84b09ea
Merge branch 'alpha' into dot-notation
dblythy Mar 7, 2023
db4f0c7
Update ParseObject.js
dblythy Mar 7, 2023
e5e0888
svelte
dblythy Mar 7, 2023
cff5e14
Merge branch 'alpha' into dot-notation
dblythy Mar 8, 2023
a744bf4
Update ParseObject.js
dblythy Mar 8, 2023
d5cfb97
Update ParseObject.js
dblythy Mar 8, 2023
a7f5387
Merge branch 'alpha' into dot-notation
mtrezza Mar 10, 2023
693a483
Update proxy.js
dblythy Mar 14, 2023
c1904fc
add copy
dblythy Mar 14, 2023
0cfc92b
Merge branch 'alpha' into dot-notation
dblythy Mar 14, 2023
4b95df3
arrays
dblythy Mar 14, 2023
9de2ac5
Update proxy.js
dblythy Jul 24, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
69 changes: 69 additions & 0 deletions integration/test/ParseObjectTest.js
Original file line number Diff line number Diff line change
Expand Up @@ -2108,6 +2108,75 @@ describe('Parse Object', () => {
});
});

it('allow binding', async () => {
const object = new Parse.Object('TestObject2');
object.bind.foo = 'bar';
await object.save();
expect(object.bind.foo).toBe('bar');
expect(object.get('foo')).toBe('bar');
expect(Object.keys(object.toJSON()).sort()).toEqual([
'createdAt',
'foo',
'objectId',
'updatedAt',
]);

const query = new Parse.Query('TestObject2');
const result = await query.get(object.id);
expect(result.bind.foo).toBe('bar');
expect(result.get('foo')).toBe('bar');
expect(result.id).toBe(object.id);
result.bind.foo = 'baz';
expect(result.get('foo')).toBe('baz');
await result.save();

const afterSave = await query.get(object.id);
expect(afterSave.bind.foo).toBe('baz');
expect(afterSave.get('foo')).toBe('baz');
});

it('allow binding on pointers', async () => {
const grandparent = new Parse.Object('DotGrandparent');
grandparent.bind.foo = 'bar1';
const parent = new Parse.Object('DotParent');
parent.bind.foo = 'bar2';
grandparent.bind.parent = parent;
const child = new Parse.Object('DotChild');
child.bind.foo = 'bar3';
parent.bind.child = child;
await Parse.Object.saveAll([child, parent, grandparent]);
expect(grandparent.bind.foo).toBe('bar1');
expect(grandparent.bind.parent.bind.foo).toBe('bar2');
expect(grandparent.bind.parent.bind.child.bind.foo).toBe('bar3');
expect(grandparent.get('foo')).toBe('bar1');
expect(grandparent.get('parent').get('foo')).toBe('bar2');
expect(grandparent.get('parent').get('child').get('foo')).toBe('bar3');
expect(Object.keys(grandparent.toJSON()).sort()).toEqual([
'createdAt',
'foo',
'objectId',
'parent',
'updatedAt',
]);
expect(Object.keys(grandparent.bind.parent.toJSON()).sort()).toEqual([
'child',
'createdAt',
'foo',
'objectId',
'updatedAt',
]);
expect(Object.keys(grandparent.bind.parent.bind.child.toJSON()).sort()).toEqual([
'createdAt',
'foo',
'objectId',
'updatedAt',
]);
const grandparentQuery = await new Parse.Query('DotGrandparent')
.include('parent', 'parent.child')
.first();
expect(grandparentQuery.bind.parent.bind.child.bind.foo).toEqual('bar3');
});

describe('allowCustomObjectId saveAll', () => {
it('can save without setting an objectId', async () => {
await reconfigureServer({ allowCustomObjectId: true });
Expand Down
23 changes: 23 additions & 0 deletions integration/test/ParseUserTest.js
Original file line number Diff line number Diff line change
Expand Up @@ -1042,6 +1042,29 @@ describe('Parse User', () => {
Parse.CoreManager.set('ENCRYPTED_KEY', null);
});

it('allow binding', async () => {
const user = new Parse.User();
const username = uuidv4();
user.bind.username = username;
user.bind.password = username;
user.bind.foo = 'bar';
await user.#();
expect(Object.keys(user.toJSON()).sort()).toEqual([
'createdAt',
'foo',
'objectId',
'sessionToken',
'updatedAt',
'username',
]);
expect(user.bind.username).toBe(username);
expect(user.bind.foo).toBe('bar');
const userFromQuery = await new Parse.Query(Parse.User).first();
expect(userFromQuery.bind.username).toBe(username);
expect(userFromQuery.bind.password).toBeUndefined();
expect(userFromQuery.bind.foo).toBe('bar');
});

it('fix GHSA-wvh7-5p38-2qfc', async () => {
Parse.User.enableUnsafeCurrentUser();
const user = new Parse.User();
Expand Down
9 changes: 3 additions & 6 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@
},
"dependencies": {
"@babel/runtime-corejs3": "7.21.0",
"deepcopy": "2.1.0",
"idb-keyval": "6.2.0",
"react-native-crypto-js": "1.0.0",
"uuid": "9.0.0",
Expand Down
33 changes: 30 additions & 3 deletions src/ParseObject.js
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ import ParseError from './ParseError';
import ParseFile from './ParseFile';
import { when, continueWhile, resolvingPromise } from './promiseUtils';
import { DEFAULT_PIN, PIN_PREFIX } from './LocalDatastoreUtils';
import proxyHandler from './proxy';

import {
opFromJSON,
Expand Down Expand Up @@ -136,6 +137,7 @@ class ParseObject {
if (toSet && !this.set(toSet, options)) {
throw new Error("Can't create an invalid Parse Object");
}
this._createProxy();
}

/**
Expand All @@ -148,6 +150,17 @@ class ParseObject {
_objCount: number;
className: string;

/**
* Bind, used for two way directonal binding using
Copy link
Member

Choose a reason for hiding this comment

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

Needs completion

*
* When using a responsive framework that supports binding to an object's keys, use `object.bind.key` for dynamic updating of a Parse.Object
*
* `object.get("key")` and `object.set("set")` is preffered for one way binding.
*
* @property {object} bind
*/
bind: AttributeMap;

/* Prototype getters / setters */

get attributes(): AttributeMap {
Expand Down Expand Up @@ -367,6 +380,7 @@ class ParseObject {
decoded.updatedAt = decoded.createdAt;
}
stateController.commitServerChanges(this._getStateIdentifier(), decoded);
this._createProxy();
}

_setExisted(existed: boolean) {
Expand All @@ -377,6 +391,10 @@ class ParseObject {
}
}

_createProxy() {
this.bind = new Proxy(this, proxyHandler);
}

_migrateId(serverId: string) {
if (this._localId && serverId) {
if (singleInstance) {
Expand Down Expand Up @@ -1098,6 +1116,8 @@ class ParseObject {
}
}
this._clearPendingOps(keysToRevert);
this._createProxy();
return this;
}

/**
Expand Down Expand Up @@ -1332,9 +1352,15 @@ class ParseObject {
}
const controller = CoreManager.getObjectController();
const unsaved = options.cascadeSave !== false ? unsavedChildren(this) : null;
return controller.save(unsaved, saveOptions).then(() => {
return controller.save(this, saveOptions);
});
return controller
.save(unsaved, saveOptions)
.then(() => {
return controller.save(this, saveOptions);
})
.then(res => {
this._createProxy();
return res;
});
}

/**
Expand Down Expand Up @@ -1972,6 +1998,7 @@ class ParseObject {
throw new Error("Can't create an invalid Parse Object");
}
}
this._createProxy();
};
if (classMap[adjustedClassName]) {
ParseObjectSubclass = classMap[adjustedClassName];
Expand Down
2 changes: 2 additions & 0 deletions src/__tests__/Parse-test.js
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,8 @@ jest.dontMock('../Parse');
jest.dontMock('../LocalDatastore');
jest.dontMock('crypto-js/aes');
jest.setMock('../EventuallyQueue', { poll: jest.fn() });
jest.dontMock('../proxy');
jest.dontMock('deepcopy');

global.indexedDB = require('./test_helpers/mockIndexedDB');
const CoreManager = require('../CoreManager');
Expand Down
72 changes: 72 additions & 0 deletions src/__tests__/ParseObject-test.js
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,8 @@ jest.dontMock('../UniqueInstanceStateController');
jest.dontMock('../unsavedChildren');
jest.dontMock('../ParseACL');
jest.dontMock('../LocalDatastore');
jest.dontMock('../proxy');
jest.dontMock('deepcopy');

jest.mock('../uuid', () => {
let value = 0;
Expand Down Expand Up @@ -2568,6 +2570,76 @@ describe('ParseObject', () => {
jest.runAllTicks();
});
});
it('can save object with dot notation', async () => {
CoreManager.getRESTController()._setXHR(
mockXHR([
{
status: 200,
response: {
objectId: 'P1',
},
},
])
);
const obj = new ParseObject('TestObject');
obj.bind.name = 'Foo';
expect(Object.keys(obj.bind)).toEqual(['name'])
await obj.save();
expect(obj.bind.name).toBe('Foo');
expect(obj.toJSON()).toEqual({ name: 'Foo', objectId: 'P1' });
expect(obj.attributes).toEqual({ name: 'Foo' });
expect(obj.get('name')).toBe('Foo');
});

it('can set and revert deep with dot notation', async () => {
CoreManager.getRESTController()._setXHR(
mockXHR([
{
status: 200,
response: { objectId: 'I1', nested: { foo: { a: 1 } } },
},
])
);
const object = await new ParseObject('Test').save();
expect(object.id).toBe('I1');
expect(object.bind.nested.foo).toEqual({ a: 1 });
object.bind.a = '123';
object.bind.nested.foo.a = 2;
expect(object.bind.nested.foo).toEqual({ a: 2 });
expect(object.dirtyKeys()).toEqual(['a', 'nested']);
object.revert('a');
expect(object.dirtyKeys()).toEqual(['nested']);
object.revert();
expect(object.bind.nested.foo).toEqual({ a: 1 });
expect(object.bind.a).toBeUndefined();
expect(object.dirtyKeys()).toEqual([]);
object.bind.nested.foo.a = 2;
expect(object.bind.nested.foo).toEqual({ a: 2 });
});

it('can delete with dot notation', async () => {
const obj = new ParseObject('TestObject');
obj.bind.name = 'Foo';
expect(obj.attributes).toEqual({ name: 'Foo' });
expect(obj.get('name')).toBe('Foo');
delete obj.bind.name;
expect(obj.op('name') instanceof ParseOp.UnsetOp).toEqual(true);
expect(obj.get('name')).toBeUndefined();
expect(obj.attributes).toEqual({});
});

it('can delete nested keys dot notation', async () => {
const obj = new ParseObject('TestObject', { name: { foo: { bar: 'a' } } });
delete obj.bind.name.foo.bar;
expect(obj.bind.name.foo).toEqual({});
});

it('can update nested array with dot notation', async () => {
const obj = new ParseObject('TestObject', { name: [{foo: { bar: 'a' } }] });
obj.bind.name[0].foo.bar = 'b';
expect(obj.get('name')).toEqual([{foo: { bar: 'b' } }]);
});

});

describe('ObjectController', () => {
Expand Down
17 changes: 17 additions & 0 deletions src/__tests__/ParseQuery-test.js
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,8 @@ jest.dontMock('../ObjectStateMutations');
jest.dontMock('../LocalDatastore');
jest.dontMock('../OfflineQuery');
jest.dontMock('../LiveQuerySubscription');
jest.dontMock('../proxy');
jest.dontMock('deepcopy');

jest.mock('../uuid', () => {
let value = 0;
Expand Down Expand Up @@ -3797,4 +3799,19 @@ describe('ParseQuery LocalDatastore', () => {
expect(subscription.sessionToken).toBe('r:test');
expect(subscription.query).toEqual(query);
});

it('can query with dot notation', async () => {
CoreManager.setQueryController({
aggregate() {},
find() {
return Promise.resolve({
results: [{ objectId: 'I1', size: 'small', name: 'Product 3' }],
});
},
});
const object = await new ParseQuery('Item').equalTo('size', 'small').first();
expect(object.id).toBe('I1');
expect(object.bind.size).toBe('small');
expect(object.bind.name).toBe('Product 3');
});
});
Loading