@coderich/autograph 0.9.2 → 0.9.6

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 CHANGED
@@ -3,6 +3,7 @@
3
3
  ## v0.9.x
4
4
  - Subscriptions API
5
5
  - postMutation no longer mutates "doc" and adds "result"
6
+ - Added onDelete defer option
6
7
 
7
8
  ## v0.8.x
8
9
  - Engine 14+
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@coderich/autograph",
3
3
  "author": "Richard Livolsi (coderich)",
4
- "version": "0.9.2",
4
+ "version": "0.9.6",
5
5
  "description": "AutoGraph",
6
6
  "keywords": [
7
7
  "graphql",
@@ -4,6 +4,7 @@ const ResultSet = require('../data/ResultSet');
4
4
  const DataLoader = require('../data/DataLoader');
5
5
  const DataTransaction = require('../data/DataTransaction');
6
6
  const QueryBuilder = require('../query/QueryBuilder');
7
+ const { createSystemEvent } = require('../service/event.service');
7
8
 
8
9
  module.exports = class Resolver {
9
10
  constructor(schema, context = {}) {
@@ -36,6 +37,31 @@ module.exports = class Resolver {
36
37
  */
37
38
  raw(model) {
38
39
  return this.toModelEntity(model).raw();
40
+ // const entity = this.toModelEntity(model);
41
+ // const driver = entity.raw();
42
+ // if (!method) return driver;
43
+
44
+ // const resolver = this;
45
+ // const crud = ['get', 'find', 'count'].indexOf(method) > -1 ? 'read' : method;
46
+ // const query = new Query({ model: entity, resolver, crud });
47
+
48
+ // return new Proxy(driver, {
49
+ // get(target, prop, rec) {
50
+ // const value = Reflect.get(target, prop, rec);
51
+
52
+ // if (typeof value === 'function') {
53
+ // return (...args) => {
54
+ // return value.bind(target)(...args).then((result) => {
55
+ // const doc = resolver.toResultSet(model, result);
56
+ // createSystemEvent('Response', { method, query: query.doc(doc) });
57
+ // return result;
58
+ // });
59
+ // };
60
+ // }
61
+
62
+ // return value;
63
+ // },
64
+ // });
39
65
  }
40
66
 
41
67
  /**
@@ -58,8 +84,8 @@ module.exports = class Resolver {
58
84
  default: {
59
85
  // This is needed in SF tests...
60
86
  const key = model.idKey();
61
- const { where } = query.toDriver();
62
- if (Object.prototype.hasOwnProperty.call(where, key) && where[key] == null) return Promise.resolve(null);
87
+ const { where, method } = query.toDriver();
88
+ if (Object.prototype.hasOwnProperty.call(where, key) && where[key] == null) return Promise.resolve(method === 'findMany' ? [] : null);
63
89
 
64
90
  //
65
91
  return this.loaders.get(`${model}`).load(query);
@@ -86,8 +112,22 @@ module.exports = class Resolver {
86
112
  return entity;
87
113
  }
88
114
 
89
- toResultSet(model, data) {
90
- return new ResultSet(new Query({ model: this.toModel(model), resolver: this }), data);
115
+ toResultSet(model, data, method) {
116
+ const crud = ['get', 'find', 'count'].indexOf(method) > -1 ? 'read' : method;
117
+ const query = new Query({ model: this.toModel(model), resolver: this, crud });
118
+ const result = new ResultSet(query, data);
119
+ return createSystemEvent('Response', {
120
+ model,
121
+ crud,
122
+ method,
123
+ result,
124
+ doc: result,
125
+ merged: result,
126
+ resolver: this,
127
+ key: `${method}${model}`,
128
+ context: this.getContext(),
129
+ query: query.doc(result).merged(result),
130
+ }, () => result);
91
131
  }
92
132
 
93
133
  // DataLoader Proxy Methods
@@ -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
  }
@@ -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;
@@ -4,6 +4,7 @@ const { map, ensureArray, keyPaths, mapPromise, toGUID, hashObject } = require('
4
4
 
5
5
  module.exports = class ResultSet {
6
6
  constructor(query, data, adjustForPagination = true) {
7
+ if (data == null) return data;
7
8
  const { resolver, model, sort, first, after, last, before } = query.toObject();
8
9
  const fields = model.getFields().filter(f => f.getName() !== 'id');
9
10
 
@@ -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}`).map(file => loadFile(file)).join('\n\n');
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`).map(file => reqFile(file)).reduce((prev, data) => {
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).filter(field => field.isBasicType() || field.isEmbedded()).map(field => `${field.getName()}: ${field.getModelRef() ? `${ucFirst(field.getDataRef())}InputWhere` : 'AutoGraphMixed'}`)}
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).filter(field => field.isBasicType() || field.isEmbedded()).map(field => `${field.getName()}: ${field.getGQLType()}`)}
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(finalResult => finalResult || result);
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