Code coverage report for lib/addons/ReactMixin.js

Statements: 89.66% (52 / 58)      Branches: 76.92% (20 / 26)      Functions: 91.67% (11 / 12)      Lines: 94.12% (48 / 51)      Ignored: none     

All files » lib/addons/ » ReactMixin.js
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166                                                                1   17   16   1 1 1   1 6 2     6                 29           17 17 17   17   1     16       1 1   1 1   1   1                                                         27 27 27 27     27 7   7 20             27 10 10   10   10   10   10   10   10 9 9     10   10   10       27           1 12    
/**
 * React Mixin
 *
 * Exports a function that creates a React component mixin. The mixin exposes
 * a Flux instance as `this.flux`. This requires that flux be passed as either
 * context or as a prop (prop takes precedence). Children also are given access
 * to flux instance as `context.flux`.
 *
 * It also adds the method `connectToStores()`, which ensures that the component
 * state stays in sync with the specified Flux stores. See the inline docs
 * of `connectToStores` for details.
 *
 * Any arguments passed to the mixin creator are passed to `connectToStores()`
 * and used as the return value of `getInitialState()`. This lets you handle
 * all of the state initialization and updates in a single place, while removing
 * the burden of manually adding and removing store listeners.
 *
 * @example
 * let Component = React.createClass({
 *   mixins: [ReactMixin({
 *     storeA: store => ({
 *       foo: store.state.a,
 *     }),
 *     storeB: store => ({
 *       bar: store.state.b,
 *     })
 *   }]
 * });
 */
 
"use strict";
 
var _interopRequire = function (obj) { return obj && obj.__esModule ? obj["default"] : obj; };
 
var _defineProperty = function (obj, key, value) { return Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); };
 
var _toArray = function (arr) { return Array.isArray(arr) ? arr : Array.from(arr); };
 
var PropTypes = require("react").PropTypes;
var Flux = require("../Flux").Flux;
var assign = _interopRequire(require("object-assign"));
 
module.exports = function () {
  for (var _len = arguments.length, args = Array(_len), _key = 0; _key < _len; _key++) {
    args[_key] = arguments[_key];
  }
 
  return {
 
    contextTypes: {
      flux: PropTypes.instanceOf(Flux) },
 
    childContextTypes: {
      flux: PropTypes.instanceOf(Flux) },
 
    getChildContext: function getChildContext() {
      return {
        flux: this.flux
      };
    },
 
    getInitialState: function getInitialState() {
      var _ref;
      this._flux_listeners = {};
      this.flux = this.props.flux || this.context.flux;
 
      if (!(this.flux instanceof Flux)) {
        // TODO: print the actual class name here
        throw new Error("ReactMixin: Could not find Flux instance. Ensure that your component " + "has either `this.context.flux` or `this.props.flux`.");
      }
 
      return (_ref = this).connectToStores.apply(_ref, _toArray(args));
    },
 
    componentWillUnmount: function componentWillUnmount() {
      for (var key in this._flux_listeners) {
        Iif (!this._flux_listeners.hasOwnProperty(key)) continue;
 
        var store = this.flux.getStore(key);
        Iif (typeof store === "undefined") continue;
 
        var listener = this._flux_listeners[key];
 
        store.removeListener("change", listener);
      }
    },
 
    /**
     * Connect component to stores, get the combined initial state, and
     * subscribe to future changes. There are three ways to call it. The
     * simplest is to pass a single store key and, optionally, a state getter.
     * The state getter is a function that takes the store as a parameter and
     * returns the state that should be passed to the component's `setState()`.
     * If no state getter is specified, the default getter is used, which simply
     * returns the entire store state.
     *
     * The second form accepts an array of store keys. With this form, every
     * store uses the default state getter.
     *
     * The last form accepts an object of store keys mapped to state getters. As
     * a shortcut, you can pass `null` as a state getter to use the default
     * state getter.
     *
     * Returns the combined initial state of all specified stores.
     *
     * This way you can write all the initialization and update logic in a single
     * location, without having to mess with adding/removing listeners.
     *
     * @type {string|array|object} stateGetterMap - map of keys to getters
     * @returns {object} Combined initial state of stores
     */
    connectToStores: function connectToStores() {
      var _this = this;
      var stateGetterMap = arguments[0] === undefined ? {} : arguments[0];
      var stateGetter = arguments[1] === undefined ? defaultStateGetter : arguments[1];
      var initialState = {};
 
      // Ensure that stateGetterMap is an object
      if (typeof stateGetterMap === "string") {
        var key = stateGetterMap;
 
        stateGetterMap = _defineProperty({}, key, stateGetter);
      } else Iif (Array.isArray(stateGetterMap)) {
        stateGetterMap.reduce(function (result, key) {
          result[key] = stateGetter;
          return result;
        }, {});
      }
 
      for (var key in stateGetterMap) {
        (function (key) {
          var store = _this.flux.getStore(key);
 
          Iif (typeof store === "undefined") throw new Error("connectToStores(): Store with key '" + key + "' does not exist.");
 
          var storeStateGetter = stateGetterMap[key];
 
          if (storeStateGetter === null) storeStateGetter = defaultStateGetter;
 
          storeStateGetter = storeStateGetter.bind(_this);
 
          var initialStoreState = storeStateGetter(store);
 
          var listener = function () {
            var state = storeStateGetter(store);
            _this.setState(state);
          };
 
          store.addListener("change", listener);
 
          assign(_this._flux_listeners, _defineProperty({}, key, listener));
 
          assign(initialState, initialStoreState);
        })(key);
      }
 
      return initialState;
    }
 
  };
};
 
function defaultStateGetter(store) {
  return store.state;
}
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbInNyYy9hZGRvbnMvUmVhY3RNaXhpbi5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUE4QkEsWUFBWSxDQUFDOzs7Ozs7OztJQUVKLFNBQVMsV0FBUSxPQUFPLEVBQXhCLFNBQVM7SUFDVCxJQUFJLFdBQVEsU0FBUyxFQUFyQixJQUFJO0lBQ04sTUFBTSwyQkFBTSxlQUFlOztpQkFHbkIsWUFBa0I7b0NBQU4sSUFBSTtBQUFKLFFBQUk7OztBQUU3QixTQUFPOztBQUVMLGdCQUFZLEVBQUU7QUFDWixVQUFJLEVBQUUsU0FBUyxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsRUFDakM7O0FBRUQscUJBQWlCLEVBQUU7QUFDakIsVUFBSSxFQUFFLFNBQVMsQ0FBQyxVQUFVLENBQUMsSUFBSSxDQUFDLEVBQ2pDOztBQUVELG1CQUFlLEVBQUEsMkJBQUc7QUFDaEIsYUFBTztBQUNMLFlBQUksRUFBRSxJQUFJLENBQUMsSUFBSTtPQUNoQixDQUFDO0tBQ0g7O0FBRUQsbUJBQWUsRUFBQSwyQkFBRzs7QUFDaEIsVUFBSSxDQUFDLGVBQWUsR0FBRyxFQUFFLENBQUM7QUFDMUIsVUFBSSxDQUFDLElBQUksR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksSUFBSSxJQUFJLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQzs7QUFFakQsVUFBSSxFQUFFLElBQUksQ0FBQyxJQUFJLFlBQVksSUFBSSxDQUFBLEFBQUMsRUFBRTs7QUFFaEMsY0FBTSxJQUFJLEtBQUssQ0FDYixnSUFDMEQsQ0FDM0QsQ0FBQztPQUNIOztBQUVELGFBQU8sUUFBQSxJQUFJLEVBQUMsZUFBZSxNQUFBLGdCQUFJLElBQUksRUFBQyxDQUFDO0tBQ3RDOztBQUVELHdCQUFvQixFQUFBLGdDQUFHO0FBQ3JCLFdBQUssSUFBSSxHQUFHLElBQUksSUFBSSxDQUFDLGVBQWUsRUFBRTtBQUNwQyxZQUFJLENBQUMsSUFBSSxDQUFDLGVBQWUsQ0FBQyxjQUFjLENBQUMsR0FBRyxDQUFDLEVBQUUsU0FBUzs7QUFFeEQsWUFBSSxLQUFLLEdBQUcsSUFBSSxDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsR0FBRyxDQUFDLENBQUM7QUFDcEMsWUFBSSxPQUFPLEtBQUssS0FBSyxXQUFXLEVBQUUsU0FBUzs7QUFFM0MsWUFBSSxRQUFRLEdBQUcsSUFBSSxDQUFDLGVBQWUsQ0FBQyxHQUFHLENBQUMsQ0FBQzs7QUFFekMsYUFBSyxDQUFDLGNBQWMsQ0FBQyxRQUFRLEVBQUUsUUFBUSxDQUFDLENBQUM7T0FDMUM7S0FDRjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUEwQkQsbUJBQWUsRUFBQSwyQkFBd0Q7O1VBQXZELGNBQWMsZ0NBQUcsRUFBRTtVQUFFLFdBQVcsZ0NBQUcsa0JBQWtCO0FBQ25FLFVBQUksWUFBWSxHQUFHLEVBQUUsQ0FBQzs7O0FBR3RCLFVBQUksT0FBTyxjQUFjLEtBQUssUUFBUSxFQUFFO0FBQ3RDLFlBQUksR0FBRyxHQUFHLGNBQWMsQ0FBQzs7QUFFekIsc0JBQWMsdUJBQ1gsR0FBRyxFQUFHLFdBQVcsQ0FDbkIsQ0FBQztPQUNILE1BQU0sSUFBSSxLQUFLLENBQUMsT0FBTyxDQUFDLGNBQWMsQ0FBQyxFQUFFO0FBQ3hDLHNCQUFjLENBQUMsTUFBTSxDQUFDLFVBQUMsTUFBTSxFQUFFLEdBQUcsRUFBSztBQUNyQyxnQkFBTSxDQUFDLEdBQUcsQ0FBQyxHQUFHLFdBQVcsQ0FBQztBQUMxQixpQkFBTyxNQUFNLENBQUM7U0FDZixFQUFFLEVBQUUsQ0FBQyxDQUFDO09BQ1I7O0FBRUQsV0FBSyxJQUFJLEdBQUcsSUFBSSxjQUFjLEVBQUU7bUJBQXZCLEdBQUc7QUFDVixjQUFJLEtBQUssR0FBRyxNQUFLLElBQUksQ0FBQyxRQUFRLENBQUMsR0FBRyxDQUFDLENBQUM7O0FBRXBDLGNBQUksT0FBTyxLQUFLLEtBQUssV0FBVyxFQUFFLE1BQU0sSUFBSSxLQUFLLHlDQUNULEdBQUcsdUJBQzFDLENBQUM7O0FBRUYsY0FBSSxnQkFBZ0IsR0FBRyxjQUFjLENBQUMsR0FBRyxDQUFDLENBQUM7O0FBRTNDLGNBQUksZ0JBQWdCLEtBQUssSUFBSSxFQUFFLGdCQUFnQixHQUFHLGtCQUFrQixDQUFDOztBQUVyRSwwQkFBZ0IsR0FBRyxnQkFBZ0IsQ0FBQyxJQUFJLE9BQU0sQ0FBQzs7QUFFL0MsY0FBSSxpQkFBaUIsR0FBRyxnQkFBZ0IsQ0FBQyxLQUFLLENBQUMsQ0FBQzs7QUFFaEQsY0FBSSxRQUFRLEdBQUcsWUFBTTtBQUNuQixnQkFBSSxLQUFLLEdBQUcsZ0JBQWdCLENBQUMsS0FBSyxDQUFDLENBQUM7QUFDcEMsa0JBQUssUUFBUSxDQUFDLEtBQUssQ0FBQyxDQUFDO1dBQ3RCLENBQUM7O0FBRUYsZUFBSyxDQUFDLFdBQVcsQ0FBQyxRQUFRLEVBQUUsUUFBUSxDQUFDLENBQUM7O0FBRXRDLGdCQUFNLENBQUMsTUFBSyxlQUFlLHNCQUN4QixHQUFHLEVBQUcsUUFBUSxFQUNmLENBQUM7O0FBRUgsZ0JBQU0sQ0FBQyxZQUFZLEVBQUUsaUJBQWlCLENBQUMsQ0FBQztXQTFCakMsR0FBRztPQTJCWDs7QUFFRCxhQUFPLFlBQVksQ0FBQztLQUNyQjs7R0FFRixDQUFBO0NBRUY7O0FBRUQsU0FBUyxrQkFBa0IsQ0FBQyxLQUFLLEVBQUU7QUFDakMsU0FBTyxLQUFLLENBQUMsS0FBSyxDQUFDO0NBQ3BCIiwiZmlsZSI6InNyYy9hZGRvbnMvUmVhY3RNaXhpbi5qcyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogUmVhY3QgTWl4aW5cbiAqXG4gKiBFeHBvcnRzIGEgZnVuY3Rpb24gdGhhdCBjcmVhdGVzIGEgUmVhY3QgY29tcG9uZW50IG1peGluLiBUaGUgbWl4aW4gZXhwb3Nlc1xuICogYSBGbHV4IGluc3RhbmNlIGFzIGB0aGlzLmZsdXhgLiBUaGlzIHJlcXVpcmVzIHRoYXQgZmx1eCBiZSBwYXNzZWQgYXMgZWl0aGVyXG4gKiBjb250ZXh0IG9yIGFzIGEgcHJvcCAocHJvcCB0YWtlcyBwcmVjZWRlbmNlKS4gQ2hpbGRyZW4gYWxzbyBhcmUgZ2l2ZW4gYWNjZXNzXG4gKiB0byBmbHV4IGluc3RhbmNlIGFzIGBjb250ZXh0LmZsdXhgLlxuICpcbiAqIEl0IGFsc28gYWRkcyB0aGUgbWV0aG9kIGBjb25uZWN0VG9TdG9yZXMoKWAsIHdoaWNoIGVuc3VyZXMgdGhhdCB0aGUgY29tcG9uZW50XG4gKiBzdGF0ZSBzdGF5cyBpbiBzeW5jIHdpdGggdGhlIHNwZWNpZmllZCBGbHV4IHN0b3Jlcy4gU2VlIHRoZSBpbmxpbmUgZG9jc1xuICogb2YgYGNvbm5lY3RUb1N0b3Jlc2AgZm9yIGRldGFpbHMuXG4gKlxuICogQW55IGFyZ3VtZW50cyBwYXNzZWQgdG8gdGhlIG1peGluIGNyZWF0b3IgYXJlIHBhc3NlZCB0byBgY29ubmVjdFRvU3RvcmVzKClgXG4gKiBhbmQgdXNlZCBhcyB0aGUgcmV0dXJuIHZhbHVlIG9mIGBnZXRJbml0aWFsU3RhdGUoKWAuIFRoaXMgbGV0cyB5b3UgaGFuZGxlXG4gKiBhbGwgb2YgdGhlIHN0YXRlIGluaXRpYWxpemF0aW9uIGFuZCB1cGRhdGVzIGluIGEgc2luZ2xlIHBsYWNlLCB3aGlsZSByZW1vdmluZ1xuICogdGhlIGJ1cmRlbiBvZiBtYW51YWxseSBhZGRpbmcgYW5kIHJlbW92aW5nIHN0b3JlIGxpc3RlbmVycy5cbiAqXG4gKiBAZXhhbXBsZVxuICogbGV0IENvbXBvbmVudCA9IFJlYWN0LmNyZWF0ZUNsYXNzKHtcbiAqICAgbWl4aW5zOiBbUmVhY3RNaXhpbih7XG4gKiAgICAgc3RvcmVBOiBzdG9yZSA9PiAoe1xuICogICAgICAgZm9vOiBzdG9yZS5zdGF0ZS5hLFxuICogICAgIH0pLFxuICogICAgIHN0b3JlQjogc3RvcmUgPT4gKHtcbiAqICAgICAgIGJhcjogc3RvcmUuc3RhdGUuYixcbiAqICAgICB9KVxuICogICB9XVxuICogfSk7XG4gKi9cblxuJ3VzZSBzdHJpY3QnO1xuXG5pbXBvcnQgeyBQcm9wVHlwZXMgfSBmcm9tICdyZWFjdCc7XG5pbXBvcnQgeyBGbHV4IH0gZnJvbSAnLi4vRmx1eCc7XG5pbXBvcnQgYXNzaWduIGZyb20gJ29iamVjdC1hc3NpZ24nO1xuXG5cbmV4cG9ydCBkZWZhdWx0IGZ1bmN0aW9uKC4uLmFyZ3MpIHtcblxuICByZXR1cm4ge1xuXG4gICAgY29udGV4dFR5cGVzOiB7XG4gICAgICBmbHV4OiBQcm9wVHlwZXMuaW5zdGFuY2VPZihGbHV4KSxcbiAgICB9LFxuXG4gICAgY2hpbGRDb250ZXh0VHlwZXM6IHtcbiAgICAgIGZsdXg6IFByb3BUeXBlcy5pbnN0YW5jZU9mKEZsdXgpLFxuICAgIH0sXG5cbiAgICBnZXRDaGlsZENvbnRleHQoKSB7XG4gICAgICByZXR1cm4ge1xuICAgICAgICBmbHV4OiB0aGlzLmZsdXhcbiAgICAgIH07XG4gICAgfSxcblxuICAgIGdldEluaXRpYWxTdGF0ZSgpIHtcbiAgICAgIHRoaXMuX2ZsdXhfbGlzdGVuZXJzID0ge307XG4gICAgICB0aGlzLmZsdXggPSB0aGlzLnByb3BzLmZsdXggfHwgdGhpcy5jb250ZXh0LmZsdXg7XG5cbiAgICAgIGlmICghKHRoaXMuZmx1eCBpbnN0YW5jZW9mIEZsdXgpKSB7XG4gICAgICAgIC8vIFRPRE86IHByaW50IHRoZSBhY3R1YWwgY2xhc3MgbmFtZSBoZXJlXG4gICAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgICBgUmVhY3RNaXhpbjogQ291bGQgbm90IGZpbmQgRmx1eCBpbnN0YW5jZS4gRW5zdXJlIHRoYXQgeW91ciBjb21wb25lbnQgYFxuICAgICAgICArIGBoYXMgZWl0aGVyIFxcYHRoaXMuY29udGV4dC5mbHV4XFxgIG9yIFxcYHRoaXMucHJvcHMuZmx1eFxcYC5gXG4gICAgICAgICk7XG4gICAgICB9XG5cbiAgICAgIHJldHVybiB0aGlzLmNvbm5lY3RUb1N0b3JlcyguLi5hcmdzKTtcbiAgICB9LFxuXG4gICAgY29tcG9uZW50V2lsbFVubW91bnQoKSB7XG4gICAgICBmb3IgKGxldCBrZXkgaW4gdGhpcy5fZmx1eF9saXN0ZW5lcnMpIHtcbiAgICAgICAgaWYgKCF0aGlzLl9mbHV4X2xpc3RlbmVycy5oYXNPd25Qcm9wZXJ0eShrZXkpKSBjb250aW51ZTtcblxuICAgICAgICBsZXQgc3RvcmUgPSB0aGlzLmZsdXguZ2V0U3RvcmUoa2V5KTtcbiAgICAgICAgaWYgKHR5cGVvZiBzdG9yZSA9PT0gJ3VuZGVmaW5lZCcpIGNvbnRpbnVlO1xuXG4gICAgICAgIGxldCBsaXN0ZW5lciA9IHRoaXMuX2ZsdXhfbGlzdGVuZXJzW2tleV07XG5cbiAgICAgICAgc3RvcmUucmVtb3ZlTGlzdGVuZXIoJ2NoYW5nZScsIGxpc3RlbmVyKTtcbiAgICAgIH1cbiAgICB9LFxuXG4gICAgLyoqXG4gICAgICogQ29ubmVjdCBjb21wb25lbnQgdG8gc3RvcmVzLCBnZXQgdGhlIGNvbWJpbmVkIGluaXRpYWwgc3RhdGUsIGFuZFxuICAgICAqIHN1YnNjcmliZSB0byBmdXR1cmUgY2hhbmdlcy4gVGhlcmUgYXJlIHRocmVlIHdheXMgdG8gY2FsbCBpdC4gVGhlXG4gICAgICogc2ltcGxlc3QgaXMgdG8gcGFzcyBhIHNpbmdsZSBzdG9yZSBrZXkgYW5kLCBvcHRpb25hbGx5LCBhIHN0YXRlIGdldHRlci5cbiAgICAgKiBUaGUgc3RhdGUgZ2V0dGVyIGlzIGEgZnVuY3Rpb24gdGhhdCB0YWtlcyB0aGUgc3RvcmUgYXMgYSBwYXJhbWV0ZXIgYW5kXG4gICAgICogcmV0dXJucyB0aGUgc3RhdGUgdGhhdCBzaG91bGQgYmUgcGFzc2VkIHRvIHRoZSBjb21wb25lbnQncyBgc2V0U3RhdGUoKWAuXG4gICAgICogSWYgbm8gc3RhdGUgZ2V0dGVyIGlzIHNwZWNpZmllZCwgdGhlIGRlZmF1bHQgZ2V0dGVyIGlzIHVzZWQsIHdoaWNoIHNpbXBseVxuICAgICAqIHJldHVybnMgdGhlIGVudGlyZSBzdG9yZSBzdGF0ZS5cbiAgICAgKlxuICAgICAqIFRoZSBzZWNvbmQgZm9ybSBhY2NlcHRzIGFuIGFycmF5IG9mIHN0b3JlIGtleXMuIFdpdGggdGhpcyBmb3JtLCBldmVyeVxuICAgICAqIHN0b3JlIHVzZXMgdGhlIGRlZmF1bHQgc3RhdGUgZ2V0dGVyLlxuICAgICAqXG4gICAgICogVGhlIGxhc3QgZm9ybSBhY2NlcHRzIGFuIG9iamVjdCBvZiBzdG9yZSBrZXlzIG1hcHBlZCB0byBzdGF0ZSBnZXR0ZXJzLiBBc1xuICAgICAqIGEgc2hvcnRjdXQsIHlvdSBjYW4gcGFzcyBgbnVsbGAgYXMgYSBzdGF0ZSBnZXR0ZXIgdG8gdXNlIHRoZSBkZWZhdWx0XG4gICAgICogc3RhdGUgZ2V0dGVyLlxuICAgICAqXG4gICAgICogUmV0dXJucyB0aGUgY29tYmluZWQgaW5pdGlhbCBzdGF0ZSBvZiBhbGwgc3BlY2lmaWVkIHN0b3Jlcy5cbiAgICAgKlxuICAgICAqIFRoaXMgd2F5IHlvdSBjYW4gd3JpdGUgYWxsIHRoZSBpbml0aWFsaXphdGlvbiBhbmQgdXBkYXRlIGxvZ2ljIGluIGEgc2luZ2xlXG4gICAgICogbG9jYXRpb24sIHdpdGhvdXQgaGF2aW5nIHRvIG1lc3Mgd2l0aCBhZGRpbmcvcmVtb3ZpbmcgbGlzdGVuZXJzLlxuICAgICAqXG4gICAgICogQHR5cGUge3N0cmluZ3xhcnJheXxvYmplY3R9IHN0YXRlR2V0dGVyTWFwIC0gbWFwIG9mIGtleXMgdG8gZ2V0dGVyc1xuICAgICAqIEByZXR1cm5zIHtvYmplY3R9IENvbWJpbmVkIGluaXRpYWwgc3RhdGUgb2Ygc3RvcmVzXG4gICAgICovXG4gICAgY29ubmVjdFRvU3RvcmVzKHN0YXRlR2V0dGVyTWFwID0ge30sIHN0YXRlR2V0dGVyID0gZGVmYXVsdFN0YXRlR2V0dGVyKSB7XG4gICAgICBsZXQgaW5pdGlhbFN0YXRlID0ge307XG5cbiAgICAgIC8vIEVuc3VyZSB0aGF0IHN0YXRlR2V0dGVyTWFwIGlzIGFuIG9iamVjdFxuICAgICAgaWYgKHR5cGVvZiBzdGF0ZUdldHRlck1hcCA9PT0gJ3N0cmluZycpIHtcbiAgICAgICAgbGV0IGtleSA9IHN0YXRlR2V0dGVyTWFwO1xuXG4gICAgICAgIHN0YXRlR2V0dGVyTWFwID0ge1xuICAgICAgICAgIFtrZXldOiBzdGF0ZUdldHRlcixcbiAgICAgICAgfTtcbiAgICAgIH0gZWxzZSBpZiAoQXJyYXkuaXNBcnJheShzdGF0ZUdldHRlck1hcCkpIHtcbiAgICAgICAgc3RhdGVHZXR0ZXJNYXAucmVkdWNlKChyZXN1bHQsIGtleSkgPT4ge1xuICAgICAgICAgIHJlc3VsdFtrZXldID0gc3RhdGVHZXR0ZXI7XG4gICAgICAgICAgcmV0dXJuIHJlc3VsdDtcbiAgICAgICAgfSwge30pO1xuICAgICAgfVxuXG4gICAgICBmb3IgKGxldCBrZXkgaW4gc3RhdGVHZXR0ZXJNYXApIHtcbiAgICAgICAgbGV0IHN0b3JlID0gdGhpcy5mbHV4LmdldFN0b3JlKGtleSk7XG5cbiAgICAgICAgaWYgKHR5cGVvZiBzdG9yZSA9PT0gJ3VuZGVmaW5lZCcpIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgICBgY29ubmVjdFRvU3RvcmVzKCk6IFN0b3JlIHdpdGgga2V5ICcke2tleX0nIGRvZXMgbm90IGV4aXN0LmBcbiAgICAgICAgKTtcblxuICAgICAgICBsZXQgc3RvcmVTdGF0ZUdldHRlciA9IHN0YXRlR2V0dGVyTWFwW2tleV07XG5cbiAgICAgICAgaWYgKHN0b3JlU3RhdGVHZXR0ZXIgPT09IG51bGwpIHN0b3JlU3RhdGVHZXR0ZXIgPSBkZWZhdWx0U3RhdGVHZXR0ZXI7XG5cbiAgICAgICAgc3RvcmVTdGF0ZUdldHRlciA9IHN0b3JlU3RhdGVHZXR0ZXIuYmluZCh0aGlzKTtcblxuICAgICAgICBsZXQgaW5pdGlhbFN0b3JlU3RhdGUgPSBzdG9yZVN0YXRlR2V0dGVyKHN0b3JlKTtcblxuICAgICAgICBsZXQgbGlzdGVuZXIgPSAoKSA9PiB7XG4gICAgICAgICAgbGV0IHN0YXRlID0gc3RvcmVTdGF0ZUdldHRlcihzdG9yZSk7XG4gICAgICAgICAgdGhpcy5zZXRTdGF0ZShzdGF0ZSk7XG4gICAgICAgIH07XG5cbiAgICAgICAgc3RvcmUuYWRkTGlzdGVuZXIoJ2NoYW5nZScsIGxpc3RlbmVyKTtcblxuICAgICAgICBhc3NpZ24odGhpcy5fZmx1eF9saXN0ZW5lcnMsIHtcbiAgICAgICAgICBba2V5XTogbGlzdGVuZXIsXG4gICAgICAgIH0pO1xuXG4gICAgICAgIGFzc2lnbihpbml0aWFsU3RhdGUsIGluaXRpYWxTdG9yZVN0YXRlKTtcbiAgICAgIH1cblxuICAgICAgcmV0dXJuIGluaXRpYWxTdGF0ZTtcbiAgICB9XG5cbiAgfVxuXG59O1xuXG5mdW5jdGlvbiBkZWZhdWx0U3RhdGVHZXR0ZXIoc3RvcmUpIHtcbiAgcmV0dXJuIHN0b3JlLnN0YXRlO1xufVxuIl19