@coderich/autograph 0.9.1 → 0.9.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +1 -0
- package/package.json +1 -1
- package/src/core/Resolver.js +5 -5
- package/src/core/Schema.js +2 -2
- package/src/data/DataService.js +12 -0
- package/src/graphql/ast/Schema.js +3 -3
- package/src/graphql/extension/api.js +17 -17
- package/src/graphql/extension/framework.js +1 -1
- package/src/query/QueryService.js +5 -0
- package/src/service/event.service.js +8 -8
package/CHANGELOG.md
CHANGED
package/package.json
CHANGED
package/src/core/Resolver.js
CHANGED
|
@@ -10,7 +10,7 @@ module.exports = class Resolver {
|
|
|
10
10
|
this.models = schema.getModels();
|
|
11
11
|
this.schema = schema;
|
|
12
12
|
this.context = context;
|
|
13
|
-
this.loaders = this.models.reduce((prev, model) => prev.set(model
|
|
13
|
+
this.loaders = this.models.reduce((prev, model) => prev.set(`${model}`, new DataLoader(this, model)), new Map());
|
|
14
14
|
|
|
15
15
|
//
|
|
16
16
|
this.getSchema = () => this.schema;
|
|
@@ -58,11 +58,11 @@ module.exports = class Resolver {
|
|
|
58
58
|
default: {
|
|
59
59
|
// This is needed in SF tests...
|
|
60
60
|
const key = model.idKey();
|
|
61
|
-
const { where } = query.toDriver();
|
|
62
|
-
if (Object.prototype.hasOwnProperty.call(where, key) && where[key] == null) return Promise.resolve(null);
|
|
61
|
+
const { where, method } = query.toDriver();
|
|
62
|
+
if (Object.prototype.hasOwnProperty.call(where, key) && where[key] == null) return Promise.resolve(method === 'findMany' ? [] : null);
|
|
63
63
|
|
|
64
64
|
//
|
|
65
|
-
return this.loaders.get(model).load(query);
|
|
65
|
+
return this.loaders.get(`${model}`).load(query);
|
|
66
66
|
}
|
|
67
67
|
}
|
|
68
68
|
}
|
|
@@ -92,7 +92,7 @@ module.exports = class Resolver {
|
|
|
92
92
|
|
|
93
93
|
// DataLoader Proxy Methods
|
|
94
94
|
clear(model) {
|
|
95
|
-
this.loaders.get(
|
|
95
|
+
this.loaders.get(`${model}`).clearAll();
|
|
96
96
|
return this;
|
|
97
97
|
}
|
|
98
98
|
|
package/src/core/Schema.js
CHANGED
|
@@ -50,8 +50,8 @@ module.exports = class extends Schema {
|
|
|
50
50
|
this.modelsByKey = this.models.reduce((prev, model) => Object.assign(prev, { [model.getKey()]: model }), {});
|
|
51
51
|
}
|
|
52
52
|
|
|
53
|
-
loadDir(dir) {
|
|
54
|
-
super.loadDir(dir);
|
|
53
|
+
loadDir(dir, options) {
|
|
54
|
+
super.loadDir(dir, options);
|
|
55
55
|
this.createModels();
|
|
56
56
|
return this;
|
|
57
57
|
}
|
package/src/data/DataService.js
CHANGED
|
@@ -79,6 +79,18 @@ exports.spliceEmbeddedArray = async (query, doc, key, from, to) => {
|
|
|
79
79
|
|
|
80
80
|
// Pull
|
|
81
81
|
if (from) {
|
|
82
|
+
// key.split('.').reduce((prev, segment, i, arr) => {
|
|
83
|
+
// if (prev == null) return prev;
|
|
84
|
+
|
|
85
|
+
// return map(prev, (data) => {
|
|
86
|
+
// const $data = data[segment];
|
|
87
|
+
// if (i < (arr.length - 1)) return $data;
|
|
88
|
+
// remove($data, el => ($from || from).find(val => objectContaining(el, val)));
|
|
89
|
+
// return $data;
|
|
90
|
+
// });
|
|
91
|
+
// }, doc);
|
|
92
|
+
|
|
93
|
+
// return doc;
|
|
82
94
|
const data = { [key]: get(doc, key) || [] };
|
|
83
95
|
remove(data[key], el => $from.find(val => objectContaining(el, val)));
|
|
84
96
|
return data;
|
|
@@ -106,12 +106,12 @@ module.exports = class Schema extends Node {
|
|
|
106
106
|
return this.schema.context;
|
|
107
107
|
}
|
|
108
108
|
|
|
109
|
-
loadDir(dir) {
|
|
109
|
+
loadDir(dir, options) {
|
|
110
110
|
// Typedefs
|
|
111
|
-
const typeDefs = Glob.sync(`${dir}/**/*.{gql,graphql}
|
|
111
|
+
const typeDefs = Glob.sync(`${dir}/**/*.{gql,graphql}`, options).map(file => loadFile(file)).join('\n\n');
|
|
112
112
|
|
|
113
113
|
// Possibly full schema definitions
|
|
114
|
-
const schema = Glob.sync(`${dir}/**/*.js
|
|
114
|
+
const schema = Glob.sync(`${dir}/**/*.js`, options).map(file => reqFile(file)).reduce((prev, data) => {
|
|
115
115
|
return Merge(prev, data);
|
|
116
116
|
}, {
|
|
117
117
|
typeDefs: typeDefs.length ? typeDefs : undefined,
|
|
@@ -70,12 +70,27 @@ module.exports = (schema) => {
|
|
|
70
70
|
`),
|
|
71
71
|
|
|
72
72
|
...entityModels.filter(model => model.hasGQLScope('s')).map(model => `
|
|
73
|
+
input ${model.getName()}SubscriptionInputFilter {
|
|
74
|
+
when: [SubscriptionWhenEnum!]! = [preEvent, postEvent]
|
|
75
|
+
where: ${model.getName()}SubscriptionInputWhere! = {}
|
|
76
|
+
}
|
|
77
|
+
|
|
73
78
|
input ${model.getName()}SubscriptionInputWhere {
|
|
74
|
-
${getGQLWhereFields(model).
|
|
79
|
+
${getGQLWhereFields(model).map(field => `${field.getName()}: ${field.getModelRef() && !field.isFKReference() ? `${ucFirst(field.getDataRef())}InputWhere` : 'AutoGraphMixed'}`)}
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
type ${model.getName()}SubscriptionPayload {
|
|
83
|
+
event: ${model.getName()}SubscriptionPayloadEvent
|
|
84
|
+
query: ${model.getName()}SubscriptionQuery
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
type ${model.getName()}SubscriptionPayloadEvent {
|
|
88
|
+
crud: SubscriptionCrudEnum!
|
|
89
|
+
data: ${model.getName()}SubscriptionPayloadEventData!
|
|
75
90
|
}
|
|
76
91
|
|
|
77
92
|
type ${model.getName()}SubscriptionPayloadEventData {
|
|
78
|
-
${getGQLWhereFields(model).
|
|
93
|
+
${getGQLWhereFields(model).map(field => `${field.getName()}: ${field.isFKReference() ? 'ID' : field.getGQLType()}`)}
|
|
79
94
|
}
|
|
80
95
|
|
|
81
96
|
interface ${model.getName()}SubscriptionQuery {
|
|
@@ -89,21 +104,6 @@ module.exports = (schema) => {
|
|
|
89
104
|
type ${model.getName()}Update implements ${model.getName()}SubscriptionQuery {
|
|
90
105
|
${model.getFields().filter(field => field.hasGQLScope('r')).map(field => `${field.getName()}: ${field.getPayloadType()}`)}
|
|
91
106
|
}
|
|
92
|
-
|
|
93
|
-
type ${model.getName()}SubscriptionPayloadEvent {
|
|
94
|
-
crud: SubscriptionCrudEnum!
|
|
95
|
-
data: ${model.getName()}SubscriptionPayloadEventData!
|
|
96
|
-
}
|
|
97
|
-
|
|
98
|
-
type ${model.getName()}SubscriptionPayload {
|
|
99
|
-
event: ${model.getName()}SubscriptionPayloadEvent
|
|
100
|
-
query: ${model.getName()}SubscriptionQuery
|
|
101
|
-
}
|
|
102
|
-
|
|
103
|
-
input ${model.getName()}SubscriptionInputFilter {
|
|
104
|
-
when: [SubscriptionWhenEnum!]! = [preEvent, postEvent]
|
|
105
|
-
where: ${model.getName()}SubscriptionInputWhere! = {}
|
|
106
|
-
}
|
|
107
107
|
`),
|
|
108
108
|
|
|
109
109
|
...spliceModels.map(model => `
|
|
@@ -11,7 +11,7 @@ module.exports = (schema) => {
|
|
|
11
11
|
enum AutoGraphTransformEnum { ${Object.keys(Transformer.getInstances()).join(' ')} }
|
|
12
12
|
enum AutoGraphAuthzEnum { private protected public }
|
|
13
13
|
enum AutoGraphValueScopeEnum { self context }
|
|
14
|
-
enum AutoGraphOnDeleteEnum { cascade nullify restrict }
|
|
14
|
+
enum AutoGraphOnDeleteEnum { cascade nullify restrict defer }
|
|
15
15
|
enum AutoGraphIndexEnum { unique }
|
|
16
16
|
|
|
17
17
|
directive @model(
|
|
@@ -111,6 +111,11 @@ exports.resolveReferentialIntegrity = (query) => {
|
|
|
111
111
|
txn.match(ref).where($where).flags(flags).count().then(count => (count ? reject(new Error('Restricted')) : count));
|
|
112
112
|
break;
|
|
113
113
|
}
|
|
114
|
+
case 'defer': {
|
|
115
|
+
// Defer to the embedded object
|
|
116
|
+
// Marks the field as an onDelete candidate otherwise it (and the embedded object) will get skipped
|
|
117
|
+
break;
|
|
118
|
+
}
|
|
114
119
|
default: throw new Error(`Unknown onDelete operator: '${op}'`);
|
|
115
120
|
}
|
|
116
121
|
});
|
|
@@ -13,11 +13,11 @@ const systemEvent = new EventEmitter().setMaxListeners(100).on('system', async (
|
|
|
13
13
|
|
|
14
14
|
//
|
|
15
15
|
exports.createSystemEvent = (name, mixed = {}, thunk = () => {}) => {
|
|
16
|
-
let event;
|
|
17
|
-
let middleware;
|
|
16
|
+
let event = mixed;
|
|
17
|
+
let middleware = () => Promise.resolve();
|
|
18
18
|
const type = ucFirst(name);
|
|
19
19
|
|
|
20
|
-
if (name !== 'Setup') {
|
|
20
|
+
if (name !== 'Setup' && name !== 'Response') {
|
|
21
21
|
const { method, query } = mixed;
|
|
22
22
|
const { resolver, model, meta, doc, id, input, sort, merged, native, root, crud } = query.toObject();
|
|
23
23
|
|
|
@@ -49,18 +49,18 @@ exports.createSystemEvent = (name, mixed = {}, thunk = () => {}) => {
|
|
|
49
49
|
|
|
50
50
|
resolve();
|
|
51
51
|
});
|
|
52
|
-
} else {
|
|
53
|
-
middleware = () => Promise.resolve();
|
|
54
|
-
event = mixed;
|
|
55
52
|
}
|
|
56
53
|
|
|
57
54
|
return systemEvent.emit('system', { type: `pre${type}`, data: event }).then((result) => {
|
|
58
55
|
if (result !== undefined) return result; // Allowing middleware to dictate result
|
|
59
56
|
return middleware().then(thunk);
|
|
60
57
|
}).then((result) => {
|
|
61
|
-
// event.doc = result; // You do actually need this...
|
|
62
58
|
event.result = result;
|
|
63
|
-
return systemEvent.emit('system', { type: `post${type}`, data: event }).then(
|
|
59
|
+
return systemEvent.emit('system', { type: `post${type}`, data: event }).then((postResult = result) => postResult);
|
|
60
|
+
}).then((result) => {
|
|
61
|
+
if (name === 'Response') return result;
|
|
62
|
+
event.result = result;
|
|
63
|
+
return exports.createSystemEvent('Response', event, (finalResult = result) => finalResult);
|
|
64
64
|
});
|
|
65
65
|
};
|
|
66
66
|
|