1 | /*
|
---|
2 | MIT License http://www.opensource.org/licenses/mit-license.php
|
---|
3 | Author Tobias Koppers @sokra
|
---|
4 | */
|
---|
5 |
|
---|
6 | "use strict";
|
---|
7 |
|
---|
8 | const makeSerializable = require("./makeSerializable.js");
|
---|
9 |
|
---|
10 | /**
|
---|
11 | * @template T
|
---|
12 | * @param {Set<T>} targetSet set where items should be added
|
---|
13 | * @param {Set<Iterable<T>>} toMerge iterables to be merged
|
---|
14 | * @returns {void}
|
---|
15 | */
|
---|
16 | const merge = (targetSet, toMerge) => {
|
---|
17 | for (const set of toMerge) {
|
---|
18 | for (const item of set) {
|
---|
19 | targetSet.add(item);
|
---|
20 | }
|
---|
21 | }
|
---|
22 | };
|
---|
23 |
|
---|
24 | /**
|
---|
25 | * @template T
|
---|
26 | * @param {Set<Iterable<T>>} targetSet set where iterables should be added
|
---|
27 | * @param {Array<LazySet<T>>} toDeepMerge lazy sets to be flattened
|
---|
28 | * @returns {void}
|
---|
29 | */
|
---|
30 | const flatten = (targetSet, toDeepMerge) => {
|
---|
31 | for (const set of toDeepMerge) {
|
---|
32 | if (set._set.size > 0) targetSet.add(set._set);
|
---|
33 | if (set._needMerge) {
|
---|
34 | for (const mergedSet of set._toMerge) {
|
---|
35 | targetSet.add(mergedSet);
|
---|
36 | }
|
---|
37 | flatten(targetSet, set._toDeepMerge);
|
---|
38 | }
|
---|
39 | }
|
---|
40 | };
|
---|
41 |
|
---|
42 | /**
|
---|
43 | * Like Set but with an addAll method to eventually add items from another iterable.
|
---|
44 | * Access methods make sure that all delayed operations are executed.
|
---|
45 | * Iteration methods deopts to normal Set performance until clear is called again (because of the chance of modifications during iteration).
|
---|
46 | * @template T
|
---|
47 | */
|
---|
48 | class LazySet {
|
---|
49 | /**
|
---|
50 | * @param {Iterable<T>=} iterable init iterable
|
---|
51 | */
|
---|
52 | constructor(iterable) {
|
---|
53 | /** @type {Set<T>} */
|
---|
54 | this._set = new Set(iterable);
|
---|
55 | /** @type {Set<Iterable<T>>} */
|
---|
56 | this._toMerge = new Set();
|
---|
57 | /** @type {Array<LazySet<T>>} */
|
---|
58 | this._toDeepMerge = [];
|
---|
59 | this._needMerge = false;
|
---|
60 | this._deopt = false;
|
---|
61 | }
|
---|
62 |
|
---|
63 | _flatten() {
|
---|
64 | flatten(this._toMerge, this._toDeepMerge);
|
---|
65 | this._toDeepMerge.length = 0;
|
---|
66 | }
|
---|
67 |
|
---|
68 | _merge() {
|
---|
69 | this._flatten();
|
---|
70 | merge(this._set, this._toMerge);
|
---|
71 | this._toMerge.clear();
|
---|
72 | this._needMerge = false;
|
---|
73 | }
|
---|
74 |
|
---|
75 | _isEmpty() {
|
---|
76 | return (
|
---|
77 | this._set.size === 0 &&
|
---|
78 | this._toMerge.size === 0 &&
|
---|
79 | this._toDeepMerge.length === 0
|
---|
80 | );
|
---|
81 | }
|
---|
82 |
|
---|
83 | get size() {
|
---|
84 | if (this._needMerge) this._merge();
|
---|
85 | return this._set.size;
|
---|
86 | }
|
---|
87 |
|
---|
88 | /**
|
---|
89 | * @param {T} item an item
|
---|
90 | * @returns {LazySet<T>} itself
|
---|
91 | */
|
---|
92 | add(item) {
|
---|
93 | this._set.add(item);
|
---|
94 | return this;
|
---|
95 | }
|
---|
96 |
|
---|
97 | /**
|
---|
98 | * @param {Iterable<T> | LazySet<T>} iterable a immutable iterable or another immutable LazySet which will eventually be merged into the Set
|
---|
99 | * @returns {LazySet<T>} itself
|
---|
100 | */
|
---|
101 | addAll(iterable) {
|
---|
102 | if (this._deopt) {
|
---|
103 | const _set = this._set;
|
---|
104 | for (const item of iterable) {
|
---|
105 | _set.add(item);
|
---|
106 | }
|
---|
107 | } else {
|
---|
108 | if (iterable instanceof LazySet) {
|
---|
109 | if (iterable._isEmpty()) return this;
|
---|
110 | this._toDeepMerge.push(iterable);
|
---|
111 | this._needMerge = true;
|
---|
112 | if (this._toDeepMerge.length > 100000) {
|
---|
113 | this._flatten();
|
---|
114 | }
|
---|
115 | } else {
|
---|
116 | this._toMerge.add(iterable);
|
---|
117 | this._needMerge = true;
|
---|
118 | }
|
---|
119 | if (this._toMerge.size > 100000) this._merge();
|
---|
120 | }
|
---|
121 | return this;
|
---|
122 | }
|
---|
123 |
|
---|
124 | clear() {
|
---|
125 | this._set.clear();
|
---|
126 | this._toMerge.clear();
|
---|
127 | this._toDeepMerge.length = 0;
|
---|
128 | this._needMerge = false;
|
---|
129 | this._deopt = false;
|
---|
130 | }
|
---|
131 |
|
---|
132 | /**
|
---|
133 | * @param {T} value an item
|
---|
134 | * @returns {boolean} true, if the value was in the Set before
|
---|
135 | */
|
---|
136 | delete(value) {
|
---|
137 | if (this._needMerge) this._merge();
|
---|
138 | return this._set.delete(value);
|
---|
139 | }
|
---|
140 |
|
---|
141 | /**
|
---|
142 | * @returns {IterableIterator<[T, T]>} entries
|
---|
143 | */
|
---|
144 | entries() {
|
---|
145 | this._deopt = true;
|
---|
146 | if (this._needMerge) this._merge();
|
---|
147 | return this._set.entries();
|
---|
148 | }
|
---|
149 |
|
---|
150 | /**
|
---|
151 | * @param {function(T, T, Set<T>): void} callbackFn function called for each entry
|
---|
152 | * @param {any} thisArg this argument for the callbackFn
|
---|
153 | * @returns {void}
|
---|
154 | */
|
---|
155 | forEach(callbackFn, thisArg) {
|
---|
156 | this._deopt = true;
|
---|
157 | if (this._needMerge) this._merge();
|
---|
158 | // eslint-disable-next-line unicorn/no-array-for-each
|
---|
159 | this._set.forEach(callbackFn, thisArg);
|
---|
160 | }
|
---|
161 |
|
---|
162 | /**
|
---|
163 | * @param {T} item an item
|
---|
164 | * @returns {boolean} true, when the item is in the Set
|
---|
165 | */
|
---|
166 | has(item) {
|
---|
167 | if (this._needMerge) this._merge();
|
---|
168 | return this._set.has(item);
|
---|
169 | }
|
---|
170 |
|
---|
171 | /**
|
---|
172 | * @returns {IterableIterator<T>} keys
|
---|
173 | */
|
---|
174 | keys() {
|
---|
175 | this._deopt = true;
|
---|
176 | if (this._needMerge) this._merge();
|
---|
177 | return this._set.keys();
|
---|
178 | }
|
---|
179 |
|
---|
180 | /**
|
---|
181 | * @returns {IterableIterator<T>} values
|
---|
182 | */
|
---|
183 | values() {
|
---|
184 | this._deopt = true;
|
---|
185 | if (this._needMerge) this._merge();
|
---|
186 | return this._set.values();
|
---|
187 | }
|
---|
188 |
|
---|
189 | /**
|
---|
190 | * @returns {IterableIterator<T>} iterable iterator
|
---|
191 | */
|
---|
192 | [Symbol.iterator]() {
|
---|
193 | this._deopt = true;
|
---|
194 | if (this._needMerge) this._merge();
|
---|
195 | return this._set[Symbol.iterator]();
|
---|
196 | }
|
---|
197 |
|
---|
198 | /* istanbul ignore next */
|
---|
199 | get [Symbol.toStringTag]() {
|
---|
200 | return "LazySet";
|
---|
201 | }
|
---|
202 |
|
---|
203 | /**
|
---|
204 | * @param {import("../serialization/ObjectMiddleware").ObjectSerializerContext} context context
|
---|
205 | */
|
---|
206 | serialize({ write }) {
|
---|
207 | if (this._needMerge) this._merge();
|
---|
208 | write(this._set.size);
|
---|
209 | for (const item of this._set) write(item);
|
---|
210 | }
|
---|
211 |
|
---|
212 | /**
|
---|
213 | * @template T
|
---|
214 | * @param {import("../serialization/ObjectMiddleware").ObjectDeserializerContext} context context
|
---|
215 | * @returns {LazySet<T>} lazy set
|
---|
216 | */
|
---|
217 | static deserialize({ read }) {
|
---|
218 | const count = read();
|
---|
219 | const items = [];
|
---|
220 | for (let i = 0; i < count; i++) {
|
---|
221 | items.push(read());
|
---|
222 | }
|
---|
223 | return new LazySet(items);
|
---|
224 | }
|
---|
225 | }
|
---|
226 |
|
---|
227 | makeSerializable(LazySet, "webpack/lib/util/LazySet");
|
---|
228 |
|
---|
229 | module.exports = LazySet;
|
---|