123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216 |
- /*
- MIT License http://www.opensource.org/licenses/mit-license.php
- Author Tobias Koppers @sokra
- */
- "use strict";
- const makeSerializable = require("./makeSerializable.js");
- /**
- * @template T
- * @param {Set<T>} targetSet set where items should be added
- * @param {Set<Iterable<T>>} toMerge iterables to be merged
- * @returns {void}
- */
- const merge = (targetSet, toMerge) => {
- for (const set of toMerge) {
- for (const item of set) {
- targetSet.add(item);
- }
- }
- };
- /**
- * @template T
- * @param {Set<Iterable<T>>} targetSet set where iterables should be added
- * @param {Array<LazySet<T>>} toDeepMerge lazy sets to be flattened
- * @returns {void}
- */
- const flatten = (targetSet, toDeepMerge) => {
- for (const set of toDeepMerge) {
- if (set._set.size > 0) targetSet.add(set._set);
- if (set._needMerge) {
- for (const mergedSet of set._toMerge) {
- targetSet.add(mergedSet);
- }
- flatten(targetSet, set._toDeepMerge);
- }
- }
- };
- /**
- * Like Set but with an addAll method to eventually add items from another iterable.
- * Access methods make sure that all delayed operations are executed.
- * Iteration methods deopts to normal Set performance until clear is called again (because of the chance of modifications during iteration).
- * @template T
- */
- class LazySet {
- /**
- * @param {Iterable<T>=} iterable init iterable
- */
- constructor(iterable) {
- /** @type {Set<T>} */
- this._set = new Set(iterable);
- /** @type {Set<Iterable<T>>} */
- this._toMerge = new Set();
- /** @type {Array<LazySet<T>>} */
- this._toDeepMerge = [];
- this._needMerge = false;
- this._deopt = false;
- }
- _flatten() {
- flatten(this._toMerge, this._toDeepMerge);
- this._toDeepMerge.length = 0;
- }
- _merge() {
- this._flatten();
- merge(this._set, this._toMerge);
- this._toMerge.clear();
- this._needMerge = false;
- }
- _isEmpty() {
- return (
- this._set.size === 0 &&
- this._toMerge.size === 0 &&
- this._toDeepMerge.length === 0
- );
- }
- get size() {
- if (this._needMerge) this._merge();
- return this._set.size;
- }
- /**
- * @param {T} item an item
- * @returns {LazySet<T>} itself
- */
- add(item) {
- this._set.add(item);
- return this;
- }
- /**
- * @param {Iterable<T> | LazySet<T>} iterable a immutable iterable or another immutable LazySet which will eventually be merged into the Set
- * @returns {LazySet<T>} itself
- */
- addAll(iterable) {
- if (this._deopt) {
- const _set = this._set;
- for (const item of iterable) {
- _set.add(item);
- }
- } else {
- if (iterable instanceof LazySet) {
- if (iterable._isEmpty()) return this;
- this._toDeepMerge.push(iterable);
- this._needMerge = true;
- if (this._toDeepMerge.length > 100000) {
- this._flatten();
- }
- } else {
- this._toMerge.add(iterable);
- this._needMerge = true;
- }
- if (this._toMerge.size > 100000) this._merge();
- }
- return this;
- }
- clear() {
- this._set.clear();
- this._toMerge.clear();
- this._toDeepMerge.length = 0;
- this._needMerge = false;
- this._deopt = false;
- }
- /**
- * @param {T} value an item
- * @returns {boolean} true, if the value was in the Set before
- */
- delete(value) {
- if (this._needMerge) this._merge();
- return this._set.delete(value);
- }
- entries() {
- this._deopt = true;
- if (this._needMerge) this._merge();
- return this._set.entries();
- }
- /**
- * @param {function(T, T, Set<T>): void} callbackFn function called for each entry
- * @param {any} thisArg this argument for the callbackFn
- * @returns {void}
- */
- forEach(callbackFn, thisArg) {
- this._deopt = true;
- if (this._needMerge) this._merge();
- this._set.forEach(callbackFn, thisArg);
- }
- /**
- * @param {T} item an item
- * @returns {boolean} true, when the item is in the Set
- */
- has(item) {
- if (this._needMerge) this._merge();
- return this._set.has(item);
- }
- keys() {
- this._deopt = true;
- if (this._needMerge) this._merge();
- return this._set.keys();
- }
- values() {
- this._deopt = true;
- if (this._needMerge) this._merge();
- return this._set.values();
- }
- [Symbol.iterator]() {
- this._deopt = true;
- if (this._needMerge) this._merge();
- return this._set[Symbol.iterator]();
- }
- /* istanbul ignore next */
- get [Symbol.toStringTag]() {
- return "LazySet";
- }
- /**
- * @param {import("../serialization/ObjectMiddleware").ObjectSerializerContext} context context
- */
- serialize({ write }) {
- if (this._needMerge) this._merge();
- write(this._set.size);
- for (const item of this._set) write(item);
- }
- /**
- * @template T
- * @param {import("../serialization/ObjectMiddleware").ObjectDeserializerContext} context context
- * @returns {LazySet<T>} lazy set
- */
- static deserialize({ read }) {
- const count = read();
- const items = [];
- for (let i = 0; i < count; i++) {
- items.push(read());
- }
- return new LazySet(items);
- }
- }
- makeSerializable(LazySet, "webpack/lib/util/LazySet");
- module.exports = LazySet;
|