1 | /**
|
2 | * @fileoverview added by tsickle
|
3 | * @suppress {checkTypes} checked by tsc
|
4 | */
|
5 | import * as tslib_1 from "tslib";
|
6 | import { BehaviorSubject, } from 'rxjs';
|
7 | import { distinctUntilChanged, first, } from 'rxjs/operators';
|
8 | import { StateSwitch } from 'state-switch';
|
9 | import { _ModelMutationType, } from '../generated-schemas/';
|
10 | /**
|
11 | * @record
|
12 | */
|
13 | export function StoreSettings() { }
|
14 | function StoreSettings_tsickle_Closure_declarations() {
|
15 | /** @type {?} */
|
16 | StoreSettings.prototype.dataKey;
|
17 | /** @type {?} */
|
18 | StoreSettings.prototype.gqlQueryAll;
|
19 | /** @type {?} */
|
20 | StoreSettings.prototype.gqlSubscribe;
|
21 | }
|
22 | /**
|
23 | * @record
|
24 | */
|
25 | export function StoreAction() { }
|
26 | function StoreAction_tsickle_Closure_declarations() {
|
27 | /** @type {?} */
|
28 | StoreAction.prototype.type;
|
29 | /** @type {?} */
|
30 | StoreAction.prototype.node;
|
31 | }
|
32 | // unsupported: template constraints.
|
33 | // unsupported: template constraints.
|
34 | // unsupported: template constraints.
|
35 | /**
|
36 | * @abstract
|
37 | * @template T, AllItemsQuery, SubscribeItemSubscription
|
38 | */
|
39 | export class Store {
|
40 | /**
|
41 | * @param {?} db
|
42 | * @param {?} settings
|
43 | */
|
44 | constructor(db, settings) {
|
45 | this.db = db;
|
46 | this.settings = settings;
|
47 | this.log = db.log;
|
48 | this.log.verbose('Store', 'constructor()');
|
49 | this.itemList$ = new BehaviorSubject([]);
|
50 | this.itemList = this.itemList$.asObservable().pipe(distinctUntilChanged());
|
51 | this.state = new StateSwitch('Store', this.log);
|
52 | /**
|
53 | * This subscription is for all the life cycle of Store,
|
54 | * we will never need to unsubscribe it.
|
55 | */
|
56 | this.db.apollo.subscribe(apollo => this.refresh(apollo));
|
57 | }
|
58 | /**
|
59 | * @return {?}
|
60 | */
|
61 | open() {
|
62 | return tslib_1.__awaiter(this, void 0, void 0, function* () {
|
63 | this.log.verbose('Store', 'open()');
|
64 | if (!this.settings) {
|
65 | throw new Error('Store.open() need `this.settings` to be set first!');
|
66 | }
|
67 | if (!this.apollo) {
|
68 | throw new Error('Store.open() apollo not available!');
|
69 | }
|
70 | this.state.on('pending');
|
71 | const /** @type {?} */ hostieQuery = this.apollo.watchQuery({
|
72 | query: this.settings.gqlQueryAll,
|
73 | });
|
74 | this.initSubscribeToMore(hostieQuery);
|
75 | this.itemListSubscription = this.initSubscription(hostieQuery);
|
76 | // await this.initQuery()
|
77 | // await future
|
78 | });
|
79 | }
|
80 | /**
|
81 | * @param {?} apollo
|
82 | * @return {?}
|
83 | */
|
84 | refresh(apollo) {
|
85 | return tslib_1.__awaiter(this, void 0, void 0, function* () {
|
86 | this.log.verbose('Store', 'refresh(%s)', apollo && apollo.constructor.name);
|
87 | /**
|
88 | * 1. close the existing apollo if it is availble
|
89 | */
|
90 | if (this.apollo) {
|
91 | yield this.close();
|
92 | }
|
93 | this.apollo = apollo;
|
94 | /**
|
95 | * 2. reopen only if the new apollo is available
|
96 | */
|
97 | if (apollo) {
|
98 | yield this.open();
|
99 | }
|
100 | });
|
101 | }
|
102 | /**
|
103 | * @return {?}
|
104 | */
|
105 | close() {
|
106 | return tslib_1.__awaiter(this, void 0, void 0, function* () {
|
107 | this.log.verbose('Store', 'close()');
|
108 | this.state.off('pending');
|
109 | if (this.itemListSubscription) {
|
110 | this.itemListSubscription.unsubscribe();
|
111 | }
|
112 | this.state.off(true);
|
113 | });
|
114 | }
|
115 | /**
|
116 | * @param {?} itemQuery
|
117 | * @return {?}
|
118 | */
|
119 | initSubscribeToMore(itemQuery) {
|
120 | this.log.verbose('Store', 'initSubscribeToMore(itemQuery)');
|
121 | itemQuery.subscribeToMore({
|
122 | document: this.settings.gqlSubscribe,
|
123 | updateQuery: (prev, { subscriptionData }) => {
|
124 | const /** @type {?} */ data = subscriptionData.data;
|
125 | const /** @type {?} */ dataKey = this.settings.dataKey;
|
126 | // if (!data || !data[dataKey]) {
|
127 | if (!data || !(dataKey in data)) {
|
128 | return prev;
|
129 | }
|
130 | const /** @type {?} */ item = data[dataKey];
|
131 | this.log.silly('Store', 'init() subscribeToMore() updateQuery() prev=%s', JSON.stringify(prev));
|
132 | this.log.silly('Store', 'init() subscribeToMore() updateQuery() data=%s', JSON.stringify(data));
|
133 | const /** @type {?} */ node = item.node;
|
134 | const /** @type {?} */ previousValues = item.previousValues;
|
135 | const /** @type {?} */ newData = /** @type {?} */ (Object.assign({}, prev));
|
136 | newData[dataKey] = this.mutationReducer(newData[dataKey], {
|
137 | type: item.mutation,
|
138 | // MutationType: CREATED / DELETED / UPDATED
|
139 | node: node || previousValues,
|
140 | });
|
141 | return newData;
|
142 | },
|
143 | onError: error => {
|
144 | this.log.warn('Store', 'initSubscribeToMore() onError() %s', JSON.stringify(error));
|
145 | },
|
146 | });
|
147 | }
|
148 | /**
|
149 | * @param {?} itemQuery
|
150 | * @return {?}
|
151 | */
|
152 | initSubscription(itemQuery) {
|
153 | this.log.verbose('Store', 'initSubscription(itemQuery)');
|
154 | const /** @type {?} */ sub = /** @type {?} */ (itemQuery.subscribe(({ data }) => {
|
155 | this.log.silly('Store', 'initSubscription() itemQuery.subscribe() data[dataKey].length=%d', data[this.settings.dataKey].length);
|
156 | this.itemList$.next([...data[this.settings.dataKey]]);
|
157 | /**
|
158 | * issue #12
|
159 | *
|
160 | * wait subscription to be ready before open() returns
|
161 | */
|
162 | this.state.on(true);
|
163 | }));
|
164 | return sub;
|
165 | }
|
166 | /**
|
167 | * @param {?=} state
|
168 | * @param {?=} action
|
169 | * @return {?}
|
170 | */
|
171 | mutationReducer(state = [], action) {
|
172 | this.log.verbose('Store', 'mutationReducer(state.length=%d, action.type=%s)', state.length, action.type);
|
173 | switch (action.type) {
|
174 | case _ModelMutationType.CREATED:
|
175 | state.push(action.node);
|
176 | break;
|
177 | case _ModelMutationType.UPDATED:
|
178 | for (let /** @type {?} */ i = state.length; i--;) {
|
179 | if (state[i].id === action.node.id) {
|
180 | state[i] = action.node;
|
181 | break;
|
182 | }
|
183 | }
|
184 | break;
|
185 | case _ModelMutationType.DELETED:
|
186 | for (let /** @type {?} */ i = state.length; i--;) {
|
187 | if (state[i].id === action.node.id) {
|
188 | state.splice(i, 1);
|
189 | break;
|
190 | }
|
191 | }
|
192 | break;
|
193 | default:
|
194 | throw new Error('unknown action.type:' + action.type);
|
195 | }
|
196 | return state;
|
197 | }
|
198 | /**
|
199 | * @param {?} mutationType
|
200 | * @param {?} mutationDataKey
|
201 | * @return {?}
|
202 | */
|
203 | mutationUpdateFnFactory(mutationType, mutationDataKey) {
|
204 | this.log.verbose('Store', 'mutationUpdateFnFactory(mutationType=%s, mutationDataKey=%s)', mutationType, mutationDataKey);
|
205 | return (proxy, { data }) => {
|
206 | this.log.verbose('Store', 'mutationUpdateFnFactory(mutationType=%s, mutationDataKey=%s) (proxy, {data})', mutationType, mutationDataKey);
|
207 | if (!data) {
|
208 | this.log.verbose('Store', 'mutationUpdateFnFactory() (proxy, {data}) data empty???');
|
209 | return;
|
210 | }
|
211 | let /** @type {?} */ cachedData = null;
|
212 | try {
|
213 | cachedData = proxy.readQuery({ query: this.settings.gqlQueryAll });
|
214 | }
|
215 | catch (/** @type {?} */ e) {
|
216 | this.log.verbose('Store', 'mutationUpdateFnFactory(mutationType=%s, mutationDataKey=%s) (proxy, {data}), %s', mutationType, mutationDataKey, 'call proxy.readQuery() got exceptoin. it mostly like there is no query had been executed before.');
|
217 | }
|
218 | if (!cachedData) {
|
219 | this.log.verbose('Store', 'mutationUpdateFnFactory(mutationType=%s, mutationDataKey=%s) (proxy, {data})proxy.readQuery() return empty???', mutationType, mutationDataKey);
|
220 | return;
|
221 | }
|
222 | const /** @type {?} */ mutationNode = data[mutationDataKey];
|
223 | /**
|
224 | * Combinate all the data to produce a new data
|
225 | */
|
226 | const /** @type {?} */ newData = /** @type {?} */ (Object.assign({}, /** @type {?} */ (cachedData)));
|
227 | newData[this.settings.dataKey] = this.mutationReducer(newData[this.settings.dataKey], {
|
228 | type: mutationType,
|
229 | node: mutationNode,
|
230 | });
|
231 | proxy.writeQuery({ query: this.settings.gqlQueryAll, data: newData });
|
232 | };
|
233 | }
|
234 | /**
|
235 | * CRUD:
|
236 | * 1. create()
|
237 | * 2. read()
|
238 | * 3. update()
|
239 | * 4. delete()
|
240 | * @param {?} id
|
241 | * @return {?}
|
242 | */
|
243 | read(id) {
|
244 | return tslib_1.__awaiter(this, void 0, void 0, function* () {
|
245 | this.log.verbose('Store', 'read(id=%s)', id);
|
246 | yield this.state.ready();
|
247 | const /** @type {?} */ itemList = yield this.itemList.pipe(first()).toPromise();
|
248 | const /** @type {?} */ result = itemList.filter(i => i['id'] === id);
|
249 | if (result.length !== 1) {
|
250 | throw new Error(`Store.read(id=${id}) not found!`);
|
251 | }
|
252 | return result[0];
|
253 | });
|
254 | }
|
255 | }
|
256 | function Store_tsickle_Closure_declarations() {
|
257 | /** @type {?} */
|
258 | Store.prototype.state;
|
259 | /** @type {?} */
|
260 | Store.prototype.itemListSubscription;
|
261 | /** @type {?} */
|
262 | Store.prototype.apollo;
|
263 | /** @type {?} */
|
264 | Store.prototype.log;
|
265 | /** @type {?} */
|
266 | Store.prototype.itemList$;
|
267 | /** @type {?} */
|
268 | Store.prototype.itemList;
|
269 | /** @type {?} */
|
270 | Store.prototype.db;
|
271 | /** @type {?} */
|
272 | Store.prototype.settings;
|
273 | /**
|
274 | * TO BE IMPLEMENT in the sub class
|
275 | * @abstract
|
276 | * @param {?} id
|
277 | * @return {?}
|
278 | */
|
279 | Store.prototype.delete = function (id) { };
|
280 | /**
|
281 | * @abstract
|
282 | * @param {?} newItem
|
283 | * @return {?}
|
284 | */
|
285 | Store.prototype.create = function (newItem) { };
|
286 | /**
|
287 | * @abstract
|
288 | * @param {?} id
|
289 | * @param {?} props
|
290 | * @return {?}
|
291 | */
|
292 | Store.prototype.update = function (id, props) { };
|
293 | }
|
294 |
|
295 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"store.js","sourceRoot":"ng://@chatie/db/","sources":["src/store.ts"],"names":[],"mappings":"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","sourcesContent":["import { DocumentNode } from 'graphql'\nimport {\n  BehaviorSubject,\n  Observable,\n  Subscription,\n}                       from 'rxjs'\nimport {\n  distinctUntilChanged,\n  first,\n}                         from 'rxjs/operators'\n\nimport { StateSwitch }  from 'state-switch'\n\nimport {\n  _ModelMutationType,\n}                       from '../generated-schemas/'\n\nimport {\n  Apollo,\n  Db,\n  MutationUpdaterFn,\n  ObservableQuery,\n}                     from './db'\nimport {\n  log,\n}                     from './config'\n\nexport interface StoreSettings {\n  dataKey:      string,\n  gqlQueryAll:  DocumentNode,\n  gqlSubscribe: DocumentNode,\n}\n\nexport interface StoreAction {\n  type: _ModelMutationType,\n  node: any,\n}\n\nexport abstract class Store<\n  T extends { [idx: string]: any },\n  AllItemsQuery extends { [idx: string]: any },\n  SubscribeItemSubscription extends { [idx: string]: any }\n> {\n  protected state: StateSwitch\n\n  private itemListSubscription?: Subscription\n\n  protected apollo?:  Apollo\n  protected log:      typeof log\n\n  protected itemList$:    BehaviorSubject< T[] >\n  public    itemList:     Observable< T[] >\n\n  constructor(\n    protected db:       Db,\n    protected settings: StoreSettings,\n  ) {\n    this.log = db.log\n\n    this.log.verbose('Store', 'constructor()')\n\n    this.itemList$  = new BehaviorSubject< T[] >([])\n    this.itemList   = this.itemList$.asObservable().pipe(distinctUntilChanged())\n\n    this.state = new StateSwitch('Store', this.log)\n\n    /**\n     * This subscription is for all the life cycle of Store,\n     * we will never need to unsubscribe it.\n     */\n    this.db.apollo.subscribe(apollo => this.refresh(apollo))\n\n  }\n\n  private async open(): Promise<void> {\n    this.log.verbose('Store', 'open()')\n    if (!this.settings) {\n      throw new Error('Store.open() need `this.settings` to be set first!')\n    }\n\n    if (!this.apollo) {\n      throw new Error('Store.open() apollo not available!')\n    }\n\n    this.state.on('pending')\n\n    const hostieQuery = this.apollo.watchQuery<AllItemsQuery>({\n      query: this.settings.gqlQueryAll,\n    })\n\n    this.initSubscribeToMore(hostieQuery)\n    this.itemListSubscription = this.initSubscription(hostieQuery)\n\n    // await this.initQuery()\n    // await future\n  }\n\n  private async refresh(apollo: Apollo | undefined): Promise<void> {\n    this.log.verbose('Store', 'refresh(%s)', apollo && apollo.constructor.name)\n\n    /**\n     * 1. close the existing apollo if it is availble\n     */\n    if (this.apollo) {\n      await this.close()\n    }\n\n    this.apollo = apollo\n\n    /**\n     * 2. reopen only if the new apollo is available\n     */\n    if (apollo) {\n      await this.open()\n    }\n\n  }\n\n  private async close():   Promise<void> {\n    this.log.verbose('Store', 'close()')\n\n    this.state.off('pending')\n\n    if (this.itemListSubscription) {\n      this.itemListSubscription.unsubscribe()\n    }\n\n    this.state.off(true)\n\n  }\n\n  private initSubscribeToMore(itemQuery: ObservableQuery<AllItemsQuery>): void {\n    this.log.verbose('Store', 'initSubscribeToMore(itemQuery)')\n    itemQuery.subscribeToMore({\n      document: this.settings.gqlSubscribe,\n      updateQuery: (prev, { subscriptionData }) => {\n        const data: SubscribeItemSubscription = subscriptionData.data\n\n        const dataKey = this.settings.dataKey\n        // if (!data || !data[dataKey]) {\n        if (!data || !(dataKey in data)) {\n          return prev\n        }\n\n        const item = data[dataKey]\n\n        this.log.silly('Store', 'init() subscribeToMore() updateQuery() prev=%s', JSON.stringify(prev))\n        this.log.silly('Store', 'init() subscribeToMore() updateQuery() data=%s', JSON.stringify(data))\n\n        const node            = item.node\n        const previousValues  = item.previousValues\n\n        const newData = {\n          ...prev,\n        } as {\n          [idx: string]: any,\n        }\n\n        newData[dataKey] = this.mutationReducer(\n          newData[dataKey],\n          {\n            type: item.mutation,          // MutationType: CREATED / DELETED / UPDATED\n            node: node || previousValues, // when DELETE, node will be null and we use previousValues\n          },\n        )\n\n        return newData\n      },\n      onError: error => {\n        this.log.warn('Store', 'initSubscribeToMore() onError() %s', JSON.stringify(error))\n      },\n    })\n  }\n\n  private initSubscription(itemQuery: ObservableQuery<AllItemsQuery>): Subscription {\n    this.log.verbose('Store', 'initSubscription(itemQuery)')\n\n    const sub = itemQuery.subscribe(\n      ({ data }) => {\n        this.log.silly('Store', 'initSubscription() itemQuery.subscribe() data[dataKey].length=%d',\n                                      data[this.settings.dataKey].length,\n                      )\n        this.itemList$.next([...data[this.settings.dataKey]])\n\n        /**\n         * issue #12\n         *\n         * wait subscription to be ready before open() returns\n         */\n        this.state.on(true)\n      },\n    ) as Subscription\n\n    return sub\n  }\n\n  private mutationReducer(\n      state:  any[] = [],\n      action: StoreAction,\n  ): any[] {\n    this.log.verbose('Store', 'mutationReducer(state.length=%d, action.type=%s)',\n                          state.length,\n                          action.type,\n                )\n    switch (action.type) {\n      case _ModelMutationType.CREATED:\n        state.push(action.node)\n        break\n\n      case _ModelMutationType.UPDATED:\n        for (let i = state.length; i--;) {\n          if (state[i].id === action.node.id) {\n            state[i] = action.node\n            break\n          }\n        }\n        break\n\n      case _ModelMutationType.DELETED:\n        for (let i = state.length; i--;) {\n          if (state[i].id === action.node.id) {\n            state.splice(i, 1)\n            break\n          }\n        }\n        break\n\n      default:\n        throw new Error('unknown action.type:' + action.type)\n    }\n\n    return state\n\n  }\n\n  protected mutationUpdateFnFactory(\n    mutationType:     _ModelMutationType,\n    mutationDataKey:  string,\n  ): MutationUpdaterFn<T> {\n    this.log.verbose('Store', 'mutationUpdateFnFactory(mutationType=%s, mutationDataKey=%s)', mutationType, mutationDataKey)\n\n    return (proxy, { data }) => {\n      this.log.verbose('Store', 'mutationUpdateFnFactory(mutationType=%s, mutationDataKey=%s) (proxy, {data})', mutationType, mutationDataKey)\n\n      if (!data) {\n        this.log.verbose('Store', 'mutationUpdateFnFactory() (proxy, {data}) data empty???')\n        return\n      }\n\n      let cachedData: AllItemsQuery | null = null\n      try {\n        cachedData = proxy.readQuery<AllItemsQuery>({ query: this.settings.gqlQueryAll })\n      } catch (e) {\n        this.log.verbose('Store', 'mutationUpdateFnFactory(mutationType=%s, mutationDataKey=%s) (proxy, {data}), %s',\n                                  mutationType,\n                                  mutationDataKey,\n                                  'call proxy.readQuery() got exceptoin. it mostly like there is no query had been executed before.')\n      }\n\n      if (!cachedData) {\n        this.log.verbose('Store', 'mutationUpdateFnFactory(mutationType=%s, mutationDataKey=%s) (proxy, {data})proxy.readQuery() return empty???',\n                                  mutationType,\n                                  mutationDataKey,\n                        )\n        return\n      }\n\n      const mutationNode = data[mutationDataKey]\n\n      /**\n       * Combinate all the data to produce a new data\n       */\n      const newData = {\n        ...cachedData as Object,\n      } as {\n        [idx: string]: any,\n      }\n      newData[this.settings.dataKey] = this.mutationReducer(\n        newData[this.settings.dataKey],\n        {\n          type: mutationType,\n          node: mutationNode,\n        },\n      )\n\n      proxy.writeQuery({ query: this.settings.gqlQueryAll, data: newData })\n\n    }\n  }\n\n  // private async initQuery(): Promise<void> {\n  //   this.log.verbose('Store', 'initQuery()')\n  //   await this.db.apollo.query<AllHostiesQuery>({\n  //     query: GQL_QUERY_ALL_HOSTIES,\n  //   })\n  //   .then(x => x.data.allHosties)\n  //   .then(hostieList => {\n  //     const queryItemMap: HostieMap = {}\n  //     for (const hostie of hostieList) {\n  //       queryItemMap[hostie.id] = hostie\n  //     }\n  //     this.$itemMap.next(queryItemMap)\n  //   })\n  // }\n\n  /**\n   * CRUD:\n   * 1. create()\n   * 2. read()\n   * 3. update()\n   * 4. delete()\n   */\n  public async read(id: string): Promise<T> {\n    this.log.verbose('Store', 'read(id=%s)', id)\n\n    await this.state.ready()\n\n    const itemList = await this.itemList.pipe(first()).toPromise()\n\n    const result = itemList.filter(i => i['id'] === id)\n    if (result.length !== 1) {\n      throw new Error(`Store.read(id=${id}) not found!`)\n    }\n    return result[0]\n  }\n\n  /**\n   * TO BE IMPLEMENT in the sub class\n   */\n  public abstract async delete(id: string):           Promise<T>\n  public abstract async create(newItem: Partial<T>):  Promise<T>\n  public abstract async update (\n    id:     string,\n    props:  T,\n  ):                                                  Promise<T>\n\n  // search: (cond: any) => Promise<[T]>,\n}\n"]} |
\ | No newline at end of file |