Skip to content

Commit 04152e9

Browse files
authored
[WIP] Feature: eager save and eager sync (#94)
* Docs * Implementation * Tests and fixes * Transformer logic * Bugfix and build * Bugfix * build * Fixes * Fix * build * build * Fix * build * Upgrade to Vuex-ORM Core 0.32.12
1 parent 84da6ee commit 04152e9

22 files changed

+523
-139
lines changed

dist/vuex-orm-graphql.es5.js

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

dist/vuex-orm-graphql.es5.js.map

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

dist/vuex-orm-graphql.umd.js

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

dist/vuex-orm-graphql.umd.js.map

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

docs/guide/eager-loading.md

+56-3
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,13 @@
1-
# Eager Loading
1+
# Eager Loading and Saving
22

33
[[toc]]
44

5-
All `belongsTo` and `hasOne` related entities are eager loaded when `fetch` is called. All other related entities have to
6-
be added to a static field in the model called `eagerLoad` to have them eagerly loaded with fetch.
5+
6+
## Eager Loading
7+
8+
All `belongsTo`, `hasOne` and `morphOne` related records are eager loaded when `fetch` is called.
9+
All other related records have to be added to a static field in the model called `eagerLoad` to
10+
have them eagerly loaded with fetch.
711

812
Example:
913

@@ -21,3 +25,52 @@ class User extends Model {
2125
}
2226
}
2327
}
28+
```
29+
30+
## Eager Saving
31+
32+
Similar to the eager loading there is a "eager saving". When saving (via `$persist` or `$push`) a
33+
record will automatically sends all `belongsTo` related records too to the server.
34+
35+
All other related records have to be added to a static field in the model called `eagerSave` to
36+
have them eagerly saved with persist and push.
37+
38+
```javascript{4}
39+
class User extends Model {
40+
static entity = 'users';
41+
static eagerLoad = ['posts'];
42+
static eagerSave = ['posts'];
43+
44+
static fields () {
45+
return {
46+
id: this.attr(null),
47+
name: this.attr(''),
48+
49+
posts: this.hasMany(Post, 'userId')
50+
}
51+
}
52+
}
53+
```
54+
55+
56+
## Eager Syncing
57+
58+
`eagerSync` combines these two fields. Adding a relation to this array will make it eagerly loaded
59+
and saved:
60+
61+
62+
```javascript{3}
63+
class User extends Model {
64+
static entity = 'users';
65+
static eagerSync = ['posts'];
66+
67+
static fields () {
68+
return {
69+
id: this.attr(null),
70+
name: this.attr(''),
71+
72+
posts: this.hasMany(Post, 'userId')
73+
}
74+
}
75+
}
76+
```

package.json

+2-2
Original file line numberDiff line numberDiff line change
@@ -56,7 +56,7 @@
5656
"app": "latest"
5757
},
5858
"peerDependencies": {
59-
"@vuex-orm/core": "^0.31.11"
59+
"@vuex-orm/core": "^0.31.12"
6060
},
6161
"devDependencies": {
6262
"@commitlint/cli": "^7.1.2",
@@ -65,7 +65,7 @@
6565
"@types/jest": "^23.3.2",
6666
"@types/node": "^10.11.0",
6767
"@types/sinon": "^5.0.1",
68-
"@vuex-orm/core": "^0.31.11",
68+
"@vuex-orm/core": "^0.31.12",
6969
"apollo-client": "^2.2.2",
7070
"apollo-link": "^1.2.0",
7171
"apollo-link-http": "^1.3.2",

src/actions/action.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -104,7 +104,7 @@ export default class Action {
104104
* @returns {Arguments}
105105
*/
106106
static addRecordToArgs(args: Arguments, model: Model, data: Data): Arguments {
107-
args[model.singularName] = Transformer.transformOutgoingData(model, data);
107+
args[model.singularName] = Transformer.transformOutgoingData(model, data, false);
108108
return args;
109109
}
110110

@@ -121,7 +121,7 @@ export default class Action {
121121

122122
if (value instanceof context.components.Model) {
123123
const model = context.getModel(singularize(value.$self().entity));
124-
const transformedValue = Transformer.transformOutgoingData(model, value);
124+
const transformedValue = Transformer.transformOutgoingData(model, value, false);
125125
context.logger.log(
126126
"A",
127127
key,

src/actions/fetch.ts

+1
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,7 @@ export default class Fetch extends Action {
3939
filter = Transformer.transformOutgoingData(
4040
model,
4141
params.filter as Data,
42+
true,
4243
Object.keys(params.filter)
4344
);
4445
}

src/actions/query.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -39,7 +39,7 @@ export default class Query extends Action {
3939
const schema: Schema = await context.loadSchema();
4040

4141
// Filter
42-
filter = filter ? Transformer.transformOutgoingData(model, filter as Data) : {};
42+
filter = filter ? Transformer.transformOutgoingData(model, filter as Data, true) : {};
4343

4444
// Multiple?
4545
const multiple: boolean = Schema.returnsConnection(schema.getQuery(name)!);

src/graphql/query-builder.ts

+11-34
Original file line numberDiff line numberDiff line change
@@ -203,7 +203,7 @@ export default class QueryBuilder {
203203
if (isPlainObject(value) && value.__type) {
204204
// Case 2 (User!)
205205
typeOrValue = context.adapter.getInputTypeName(context.getModel(value.__type)) + "!";
206-
} else if (Array.isArray(value) && field) {
206+
} else if (value instanceof Array && field) {
207207
const arg = QueryBuilder.findSchemaFieldForArgument(key, field, model, filter);
208208

209209
/* istanbul ignore next */
@@ -353,38 +353,8 @@ export default class QueryBuilder {
353353
const context = Context.getInstance();
354354
const relationQueries: Array<string> = [];
355355

356-
model.getRelations().forEach((field: Field, name: string) => {
357-
let relatedModel: Model;
358-
let fieldAsRelation: Relation = field as Relation;
359-
360-
if (
361-
fieldAsRelation instanceof context.components.BelongsToMany ||
362-
fieldAsRelation instanceof context.components.HasMany ||
363-
fieldAsRelation instanceof context.components.HasManyThrough ||
364-
fieldAsRelation instanceof context.components.MorphedByMany ||
365-
fieldAsRelation instanceof context.components.MorphMany ||
366-
fieldAsRelation instanceof context.components.MorphOne ||
367-
fieldAsRelation instanceof context.components.MorphToMany ||
368-
fieldAsRelation instanceof context.components.HasOne
369-
) {
370-
relatedModel = context.getModel(fieldAsRelation.related.entity);
371-
} else if (
372-
fieldAsRelation instanceof context.components.BelongsTo ||
373-
fieldAsRelation instanceof context.components.HasManyBy
374-
) {
375-
relatedModel = context.getModel(fieldAsRelation.parent.entity);
376-
} else if (fieldAsRelation instanceof context.components.MorphTo) {
377-
relatedModel = context.getModel(fieldAsRelation.type);
378-
379-
/* istanbul ignore next */
380-
} else {
381-
relatedModel = context.getModel(name);
382-
383-
context.logger.log(
384-
"WARNING: unknown field type. Fallback to attribute name",
385-
fieldAsRelation
386-
);
387-
}
356+
model.getRelations().forEach((field: Relation, name: string) => {
357+
let relatedModel: Model = Model.getRelatedModel(field)!;
388358

389359
// We will ignore the field, when it's already in the path. Means: When it's already queried. However there are
390360
// cases where the model will have a relationship to itself. For example a nested category strucure where the
@@ -405,7 +375,14 @@ export default class QueryBuilder {
405375
newPath.push(relatedModel.singularName);
406376

407377
relationQueries.push(
408-
this.buildField(relatedModel, Model.isConnection(field), undefined, newPath, name, false)
378+
this.buildField(
379+
relatedModel,
380+
Model.isConnection(field as Field),
381+
undefined,
382+
newPath,
383+
name,
384+
false
385+
)
409386
);
410387
}
411388
});

src/graphql/schema.ts

+9-8
Original file line numberDiff line numberDiff line change
@@ -106,17 +106,18 @@ export default class Schema {
106106
}
107107

108108
static getTypeNameOfField(field: GraphQLField): string {
109-
const type = this.getRealType(field.type);
109+
let type = this.getRealType(field.type);
110110

111111
if (type.kind === "LIST") {
112-
return `[${type.ofType.name}]`;
113-
}
112+
while (!type.name) type = type.ofType;
113+
return `[${type.name}]`;
114+
} else {
115+
while (!type.name) type = type.ofType;
114116

115-
const name = type.name || type.ofType.name || type.ofType.ofType.name;
117+
/* istanbul ignore next */
118+
if (!type.name) throw new Error(`Can't find type name for field ${field.name}`);
116119

117-
/* istanbul ignore next */
118-
if (!name) throw new Error(`Can't find type name for field ${field.name}`);
119-
120-
return name;
120+
return type.name;
121+
}
121122
}
122123
}

0 commit comments

Comments
 (0)