Home Reference Source

packages/remote-context/src/actions/RemoteSetObjectAction.js

  1. /**
  2. * Copyright 2017 Moshe Simantov
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16.  
  17. import { trimArgumentsList } from 'remote-instance';
  18. import RemoteSetAction from './RemoteSetAction';
  19. import PropertyDescriptorAction from './PropertyDescriptorAction';
  20. import { setCachedGetter } from '../helpers/descriptors';
  21.  
  22. const ARRAY_PROTOTYPE_CODE = 1;
  23.  
  24. function toPropertiesArray(session, obj, mapFn) {
  25. return Object.keys(obj).reduce((arr, property) => {
  26. arr.push(session.dispatch(property), mapFn(obj[property], property));
  27. return arr;
  28. }, []);
  29. }
  30.  
  31. function isValidPropertiesArray(propsArr) {
  32. return Array.isArray(propsArr) && propsArr.length % 2 === 0;
  33. }
  34.  
  35. function fromPropertiesArray(session, propsArr, forEachFn) {
  36. for (let i = 0; i < propsArr.length; i += 2) {
  37. const property = session.fetch(propsArr[i]);
  38. const value = session.fetch(propsArr[i + 1]);
  39.  
  40. forEachFn(value, property);
  41. }
  42. }
  43.  
  44. export default class RemoteSetObjectAction extends RemoteSetAction {
  45. static fromSnapshot(session, reference, snapshot) {
  46. const obj = snapshot.value;
  47. if (obj === null) return null;
  48.  
  49. // Fetch order should be the same: first prototype then the descriptors
  50. const proto = session.dispatch(this.getPrototype(snapshot));
  51.  
  52. const descriptors = toPropertiesArray(
  53. session,
  54. snapshot.ownPropertyDescriptors,
  55. desc => PropertyDescriptorAction.fromPropertyDescriptor(session, desc)
  56. );
  57.  
  58. const isExtensible = Object.isExtensible(obj);
  59.  
  60. const cachedGetters = toPropertiesArray(
  61. session,
  62. snapshot.cachedGetters,
  63. cachedValue => session.dispatch(cachedValue)
  64. );
  65.  
  66. return new this(reference, descriptors, proto, isExtensible, cachedGetters);
  67. }
  68.  
  69. static getPrototype(snapshot) {
  70. const proto = snapshot.prototype;
  71.  
  72. if (proto === Object.prototype) {
  73. return null;
  74. }
  75.  
  76. if (proto === Array.prototype) {
  77. return ARRAY_PROTOTYPE_CODE;
  78. }
  79.  
  80. return proto;
  81. }
  82.  
  83. constructor(
  84. reference,
  85. descriptors = [],
  86. proto = null,
  87. isExtensible = true,
  88. cachedGetters = []
  89. ) {
  90. if (!isValidPropertiesArray(descriptors)) {
  91. throw new TypeError(`Invalid descriptors: ${descriptors}`);
  92. }
  93.  
  94. if (typeof proto !== 'object') {
  95. if (proto !== ARRAY_PROTOTYPE_CODE) {
  96. throw new TypeError('Object prototype may only be an Object or null');
  97. }
  98. }
  99.  
  100. if (typeof isExtensible !== 'boolean') {
  101. throw new TypeError('Expect "isExtensible" to be boolean');
  102. }
  103.  
  104. if (!isValidPropertiesArray(cachedGetters)) {
  105. throw new TypeError(`Invalid "cachedGetters" param: ${cachedGetters}`);
  106. }
  107.  
  108. super(reference);
  109.  
  110. this.descriptors = descriptors;
  111. this.proto = proto;
  112. this.isExtensible = isExtensible;
  113. this.cachedGetters = cachedGetters;
  114. }
  115.  
  116. // eslint-disable-next-line class-methods-use-this
  117. createInstance() {
  118. if (this.proto === ARRAY_PROTOTYPE_CODE) {
  119. return [];
  120. }
  121.  
  122. return {};
  123. }
  124.  
  125. populateInstance(session, instance) {
  126. // Fetch order the same as dispatch order (ie. first prototype then descriptors)
  127. // @see {@link RemoteSetObjectAction.fromValue}
  128.  
  129. const proto = session.fetch(this.proto);
  130. if (proto !== null && typeof proto === 'object') {
  131. Object.setPrototypeOf(instance, proto);
  132. }
  133.  
  134. fromPropertiesArray(session, this.descriptors, (descriptor, property) => {
  135. Object.defineProperty(instance, property, descriptor);
  136. });
  137.  
  138. fromPropertiesArray(session, this.cachedGetters, (value, property) => {
  139. setCachedGetter(instance, property, value);
  140. });
  141. }
  142.  
  143. toArgumentsList() {
  144. const argumentsList = [
  145. this.reference,
  146. this.descriptors,
  147. this.proto,
  148. this.isExtensible,
  149. this.cachedGetters,
  150. ];
  151. const defaults = [undefined, [], null, true, []];
  152.  
  153. return trimArgumentsList(argumentsList, defaults);
  154. }
  155. }
  156.  
  157. RemoteSetAction.types.object = RemoteSetObjectAction;