@splitsoftware/splitio 10.17.3-rc.0 → 10.17.3-rc.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (46) hide show
  1. package/client/package.json +5 -0
  2. package/es/factory/{clientSide.js → browser.js} +24 -11
  3. package/es/factory/{serverSide.js → node.js} +28 -11
  4. package/es/factory/package.json +2 -2
  5. package/es/platform/EventEmitter.js +412 -0
  6. package/es/platform/browser.js +2 -3
  7. package/es/platform/node.js +1 -2
  8. package/es/settings/{clientSide.js → browser.js} +0 -0
  9. package/es/settings/defaults/version.js +1 -1
  10. package/es/settings/{serverSide.js → node.js} +0 -0
  11. package/es/settings/package.json +2 -2
  12. package/es/settings/storage/browser.js +5 -4
  13. package/es/settings/storage/node.js +17 -9
  14. package/lib/factory/{clientSide.js → browser.js} +26 -12
  15. package/lib/factory/{serverSide.js → node.js} +30 -12
  16. package/lib/factory/package.json +2 -2
  17. package/lib/platform/EventEmitter.js +417 -0
  18. package/lib/platform/browser.js +5 -8
  19. package/lib/platform/node.js +3 -4
  20. package/lib/settings/{clientSide.js → browser.js} +0 -0
  21. package/lib/settings/defaults/version.js +1 -1
  22. package/lib/settings/{serverSide.js → node.js} +0 -0
  23. package/lib/settings/package.json +2 -2
  24. package/lib/settings/storage/browser.js +5 -4
  25. package/lib/settings/storage/node.js +17 -10
  26. package/package.json +5 -4
  27. package/server/package.json +5 -0
  28. package/src/.DS_Store +0 -0
  29. package/src/factory/{clientSide.js → browser.js} +31 -15
  30. package/src/factory/{serverSide.js → node.js} +34 -14
  31. package/src/factory/package.json +2 -2
  32. package/src/platform/EventEmitter.js +443 -0
  33. package/src/platform/browser.js +2 -3
  34. package/src/platform/node.js +1 -2
  35. package/src/settings/{clientSide.js → browser.js} +0 -0
  36. package/src/settings/defaults/version.js +1 -1
  37. package/src/settings/{serverSide.js → node.js} +0 -0
  38. package/src/settings/package.json +2 -2
  39. package/src/settings/storage/browser.js +4 -3
  40. package/src/settings/storage/node.js +15 -10
  41. package/types/client/index.d.ts +14 -0
  42. package/types/server/index.d.ts +20 -0
  43. package/types/splitio.d.ts +22 -5
  44. package/es/factory/commons.js +0 -16
  45. package/lib/factory/commons.js +0 -23
  46. package/src/factory/commons.js +0 -16
@@ -20,10 +20,9 @@ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
20
20
  See the License for the specific language governing permissions and
21
21
  limitations under the License.
22
22
  **/
23
- var STORAGE_REDIS = 'REDIS';
24
-
25
23
  function validateStorage(settings) {
26
- var mode = settings.mode,
24
+ var log = settings.log,
25
+ mode = settings.mode,
27
26
  _settings$storage = settings.storage;
28
27
  _settings$storage = _settings$storage === void 0 ? {
29
28
  type: _constants.STORAGE_MEMORY
@@ -37,17 +36,21 @@ function validateStorage(settings) {
37
36
  prefix += '.SPLITIO';
38
37
  } else {
39
38
  prefix = 'SPLITIO';
40
- } // In localhost mode we should force the user to use the MEMORY storage
41
-
39
+ } // We can have MEMORY, REDIS or an invalid storage type
42
40
 
43
- if (mode === _constants.LOCALHOST_MODE) return {
44
- type: _constants.STORAGE_MEMORY,
45
- prefix: prefix
46
- }; // In other cases we can have MEMORY or REDIS
47
41
 
48
42
  switch (type) {
49
- case STORAGE_REDIS:
43
+ case _constants.STORAGE_REDIS:
50
44
  {
45
+ // If passing REDIS storage in localhost or standalone mode, we log an error and fallback to MEMORY storage
46
+ if (mode === _constants.STANDALONE_MODE || mode === _constants.LOCALHOST_MODE) {
47
+ log.error('The provided REDIS storage is invalid for this mode. It requires consumer mode. Fallbacking into default MEMORY storage.');
48
+ return {
49
+ type: _constants.STORAGE_MEMORY,
50
+ prefix: prefix
51
+ };
52
+ }
53
+
51
54
  var host = options.host,
52
55
  port = options.port,
53
56
  db = options.db,
@@ -85,6 +88,10 @@ function validateStorage(settings) {
85
88
  case _constants.STORAGE_MEMORY:
86
89
  default:
87
90
  {
91
+ // If passing MEMORY storage in consumer mode, throw an error (no way to fallback to REDIS storage)
92
+ if (mode === _constants.CONSUMER_MODE) throw new Error('A REDIS storage is required on consumer mode'); // If passing an invalid storage type, log an error
93
+
94
+ if (type !== _constants.STORAGE_MEMORY) log.error("The provided '" + type + "' storage type is invalid. Fallbacking into default MEMORY storage.");
88
95
  return {
89
96
  type: _constants.STORAGE_MEMORY,
90
97
  prefix: prefix
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@splitsoftware/splitio",
3
- "version": "10.17.3-rc.0",
3
+ "version": "10.17.3-rc.1",
4
4
  "description": "Split SDK",
5
5
  "files": [
6
6
  "README.md",
@@ -8,9 +8,11 @@
8
8
  "LICENSE",
9
9
  "CHANGES.txt",
10
10
  "lib",
11
- "types/*.d.ts",
11
+ "types",
12
12
  "es",
13
- "src"
13
+ "src",
14
+ "client",
15
+ "server"
14
16
  ],
15
17
  "repository": "splitio/javascript-client",
16
18
  "homepage": "https://github.com/splitio/javascript-client#readme",
@@ -35,7 +37,6 @@
35
37
  "@babel/runtime": "^7.13.10",
36
38
  "@splitsoftware/splitio-commons": "1.2.0",
37
39
  "@types/google.analytics": "0.0.40",
38
- "events": "3.1.0",
39
40
  "ioredis": "^4.28.0",
40
41
  "ip": "1.1.5",
41
42
  "js-yaml": "3.13.1",
@@ -0,0 +1,5 @@
1
+ {
2
+ "main": "../lib/factory/node.js",
3
+ "module": "../es/factory/node.js",
4
+ "types": "../types/server/index.d.ts"
5
+ }
package/src/.DS_Store CHANGED
Binary file
@@ -10,45 +10,61 @@ import { impressionObserverCSFactory } from '@splitsoftware/splitio-commons/src/
10
10
  import { integrationsManagerFactory } from '@splitsoftware/splitio-commons/src/integrations/browser';
11
11
  import { __InLocalStorageMockFactory } from '@splitsoftware/splitio-commons/src/utils/settingsValidation/storage/storageCS';
12
12
  import { sdkFactory } from '@splitsoftware/splitio-commons/src/sdkFactory';
13
+ import { LOCALHOST_MODE, STORAGE_LOCALSTORAGE } from '@splitsoftware/splitio-commons/src/utils/constants';
14
+ import { shouldAddPt } from '@splitsoftware/splitio-commons/src/trackers/impressionObserver/utils';
13
15
 
14
- import { settingsFactory } from '../settings/clientSide';
15
- import { shouldAddPt } from './commons';
16
- import { platform, signalListener } from '../platform';
16
+ import { settingsFactory } from '../settings/browser';
17
+ import { platform, SignalListener } from '../platform';
17
18
 
18
19
  const syncManagerOnlineCSFactory = syncManagerOnlineFactory(pollingManagerCSFactory, pushManagerFactory);
19
20
 
21
+ function getStorage(settings) {
22
+ return settings.storage.type === STORAGE_LOCALSTORAGE ?
23
+ InLocalStorage(settings.storage)
24
+ : settings.storage.__originalType === STORAGE_LOCALSTORAGE ?
25
+ __InLocalStorageMockFactory
26
+ : InMemoryStorageCSFactory;
27
+ }
28
+
20
29
  /**
21
30
  *
22
31
  * @param {import("@splitsoftware/splitio-commons/types/types").ISettings} settings
23
32
  */
24
33
  function getModules(settings) {
25
34
 
26
- return {
35
+ const modules = {
27
36
  settings,
28
37
 
29
38
  platform,
30
39
 
31
- storageFactory: settings.storage.type === 'LOCALSTORAGE' ?
32
- InLocalStorage({
33
- prefix: settings.storage.prefix,
34
- })
35
- : settings.storage.__originalType === 'LOCALSTORAGE' ?
36
- __InLocalStorageMockFactory
37
- : InMemoryStorageCSFactory,
40
+ storageFactory: getStorage(settings),
38
41
 
39
- splitApiFactory: settings.mode === 'localhost' ? undefined : splitApiFactory,
40
- syncManagerFactory: settings.mode === 'localhost' ? settings.sync.localhostMode : syncManagerOnlineCSFactory,
42
+ splitApiFactory,
43
+
44
+ syncManagerFactory: syncManagerOnlineCSFactory,
41
45
 
42
46
  sdkManagerFactory,
47
+
43
48
  sdkClientMethodFactory: sdkClientMethodCSFactory,
44
- SignalListener: settings.mode === 'localhost' ? undefined : signalListener,
49
+
50
+ SignalListener,
51
+
45
52
  impressionListener: settings.impressionListener,
46
53
 
47
54
  integrationsManagerFactory: settings.integrations && settings.integrations.length > 0 ? integrationsManagerFactory.bind(null, settings.integrations) : undefined,
48
55
 
49
- // @TODO consider not including in debug mode?
50
56
  impressionsObserverFactory: shouldAddPt(settings) ? impressionObserverCSFactory : undefined,
51
57
  };
58
+
59
+ switch (settings.mode) {
60
+ case LOCALHOST_MODE:
61
+ modules.splitApiFactory = undefined;
62
+ modules.syncManagerFactory = settings.sync.localhostMode;
63
+ modules.SignalListener = undefined;
64
+ break;
65
+ }
66
+
67
+ return modules;
52
68
  }
53
69
 
54
70
  export function SplitFactory(config) {
@@ -8,40 +8,60 @@ import { sdkManagerFactory } from '@splitsoftware/splitio-commons/src/sdkManager
8
8
  import { sdkClientMethodFactory } from '@splitsoftware/splitio-commons/src/sdkClient/sdkClientMethod';
9
9
  import { impressionObserverSSFactory } from '@splitsoftware/splitio-commons/src/trackers/impressionObserver/impressionObserverSS';
10
10
  import { sdkFactory } from '@splitsoftware/splitio-commons/src/sdkFactory';
11
+ import { CONSUMER_MODE, LOCALHOST_MODE } from '@splitsoftware/splitio-commons/src/utils/constants';
12
+ import { shouldAddPt } from '@splitsoftware/splitio-commons/src/trackers/impressionObserver/utils';
11
13
 
12
- import { settingsFactory } from '../settings/serverSide';
13
- import { shouldAddPt } from './commons';
14
- import { platform, signalListener } from '../platform';
14
+ import { settingsFactory } from '../settings/node';
15
+ import { platform, SignalListener } from '../platform';
15
16
 
16
17
  const syncManagerOnlineSSFactory = syncManagerOnlineFactory(pollingManagerSSFactory, pushManagerFactory);
17
18
 
19
+ function getStorage(settings) {
20
+ return settings.storage.type === 'REDIS' ?
21
+ InRedisStorage(settings.storage) :
22
+ InMemoryStorageFactory;
23
+ }
24
+
18
25
  /**
19
26
  *
20
27
  * @param {import("@splitsoftware/splitio-commons/types/types").ISettings} settings
21
28
  */
22
29
  function getModules(settings) {
23
- return {
30
+
31
+ const modules = {
24
32
  settings,
25
33
 
26
34
  platform,
27
35
 
28
- storageFactory: settings.storage.type === 'REDIS' ?
29
- InRedisStorage({
30
- prefix: settings.storage.prefix,
31
- options: settings.storage.options
32
- }) :
33
- InMemoryStorageFactory,
36
+ storageFactory: getStorage(settings),
34
37
 
35
- splitApiFactory: settings.mode === 'localhost' ? undefined : splitApiFactory,
36
- syncManagerFactory: settings.storage.type === 'REDIS' ? undefined : settings.mode === 'localhost' ? settings.sync.localhostMode : syncManagerOnlineSSFactory,
38
+ splitApiFactory,
39
+
40
+ syncManagerFactory: syncManagerOnlineSSFactory,
37
41
 
38
42
  sdkManagerFactory,
39
- sdkClientMethodFactory: sdkClientMethodFactory,
40
- SignalListener: settings.mode === 'localhost' ? undefined : signalListener,
43
+
44
+ sdkClientMethodFactory,
45
+
46
+ SignalListener,
47
+
41
48
  impressionListener: settings.impressionListener,
42
49
 
43
50
  impressionsObserverFactory: shouldAddPt(settings) ? impressionObserverSSFactory : undefined,
44
51
  };
52
+
53
+ switch (settings.mode) {
54
+ case LOCALHOST_MODE:
55
+ modules.splitApiFactory = undefined;
56
+ modules.syncManagerFactory = settings.sync.localhostMode;
57
+ modules.SignalListener = undefined;
58
+ break;
59
+ case CONSUMER_MODE:
60
+ modules.syncManagerFactory = undefined;
61
+ break;
62
+ }
63
+
64
+ return modules;
45
65
  }
46
66
 
47
67
  export function SplitFactory(config) {
@@ -1,4 +1,4 @@
1
1
  {
2
- "main": "./serverSide.js",
3
- "browser": "./clientSide.js"
2
+ "main": "./node.js",
3
+ "browser": "./browser.js"
4
4
  }
@@ -0,0 +1,443 @@
1
+ // Copyright Joyent, Inc. and other Node contributors.
2
+ //
3
+ // Permission is hereby granted, free of charge, to any person obtaining a
4
+ // copy of this software and associated documentation files (the
5
+ // "Software"), to deal in the Software without restriction, including
6
+ // without limitation the rights to use, copy, modify, merge, publish,
7
+ // distribute, sublicense, and/or sell copies of the Software, and to permit
8
+ // persons to whom the Software is furnished to do so, subject to the
9
+ // following conditions:
10
+ //
11
+ // The above copyright notice and this permission notice shall be included
12
+ // in all copies or substantial portions of the Software.
13
+ //
14
+ // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
15
+ // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16
+ // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
17
+ // NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
18
+ // DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
19
+ // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
20
+ // USE OR OTHER DEALINGS IN THE SOFTWARE.
21
+
22
+ // Adaptation of "events" library (https://www.npmjs.com/package/events)
23
+ // exported as an ES module instead of CommonJS, to avoid extra configuration steps when using
24
+ // the ESM build of the SDK with tools that doesn't support CommonJS by default (e.g. Rollup).
25
+
26
+ var R = typeof Reflect === 'object' ? Reflect : null;
27
+ var ReflectApply = R && typeof R.apply === 'function'
28
+ ? R.apply
29
+ : function ReflectApply(target, receiver, args) {
30
+ return Function.prototype.apply.call(target, receiver, args);
31
+ };
32
+
33
+ var ReflectOwnKeys;
34
+ if (R && typeof R.ownKeys === 'function') {
35
+ ReflectOwnKeys = R.ownKeys;
36
+ } else if (Object.getOwnPropertySymbols) {
37
+ ReflectOwnKeys = function ReflectOwnKeys(target) {
38
+ return Object.getOwnPropertyNames(target)
39
+ .concat(Object.getOwnPropertySymbols(target));
40
+ };
41
+ } else {
42
+ ReflectOwnKeys = function ReflectOwnKeys(target) {
43
+ return Object.getOwnPropertyNames(target);
44
+ };
45
+ }
46
+
47
+ function ProcessEmitWarning(warning) {
48
+ if (console && console.warn) console.warn(warning);
49
+ }
50
+
51
+ var NumberIsNaN = Number.isNaN || function NumberIsNaN(value) {
52
+ return value !== value;
53
+ };
54
+
55
+ export function EventEmitter() {
56
+ EventEmitter.init.call(this);
57
+ }
58
+
59
+ EventEmitter.prototype._events = undefined;
60
+ EventEmitter.prototype._eventsCount = 0;
61
+ EventEmitter.prototype._maxListeners = undefined;
62
+
63
+ // By default EventEmitters will print a warning if more than 10 listeners are
64
+ // added to it. This is a useful default which helps finding memory leaks.
65
+ var defaultMaxListeners = 10;
66
+
67
+ function checkListener(listener) {
68
+ if (typeof listener !== 'function') {
69
+ throw new TypeError('The "listener" argument must be of type Function. Received type ' + typeof listener);
70
+ }
71
+ }
72
+
73
+ Object.defineProperty(EventEmitter, 'defaultMaxListeners', {
74
+ enumerable: true,
75
+ get: function() {
76
+ return defaultMaxListeners;
77
+ },
78
+ set: function(arg) {
79
+ if (typeof arg !== 'number' || arg < 0 || NumberIsNaN(arg)) {
80
+ throw new RangeError('The value of "defaultMaxListeners" is out of range. It must be a non-negative number. Received ' + arg + '.');
81
+ }
82
+ defaultMaxListeners = arg;
83
+ }
84
+ });
85
+
86
+ EventEmitter.init = function() {
87
+
88
+ if (this._events === undefined ||
89
+ this._events === Object.getPrototypeOf(this)._events) {
90
+ this._events = Object.create(null);
91
+ this._eventsCount = 0;
92
+ }
93
+
94
+ this._maxListeners = this._maxListeners || undefined;
95
+ };
96
+
97
+ // Obviously not all Emitters should be limited to 10. This function allows
98
+ // that to be increased. Set to zero for unlimited.
99
+ EventEmitter.prototype.setMaxListeners = function setMaxListeners(n) {
100
+ if (typeof n !== 'number' || n < 0 || NumberIsNaN(n)) {
101
+ throw new RangeError('The value of "n" is out of range. It must be a non-negative number. Received ' + n + '.');
102
+ }
103
+ this._maxListeners = n;
104
+ return this;
105
+ };
106
+
107
+ function _getMaxListeners(that) {
108
+ if (that._maxListeners === undefined)
109
+ return EventEmitter.defaultMaxListeners;
110
+ return that._maxListeners;
111
+ }
112
+
113
+ EventEmitter.prototype.getMaxListeners = function getMaxListeners() {
114
+ return _getMaxListeners(this);
115
+ };
116
+
117
+ EventEmitter.prototype.emit = function emit(type) {
118
+ var args = [];
119
+ for (var i = 1; i < arguments.length; i++) args.push(arguments[i]);
120
+ var doError = (type === 'error');
121
+
122
+ var events = this._events;
123
+ if (events !== undefined)
124
+ doError = (doError && events.error === undefined);
125
+ else if (!doError)
126
+ return false;
127
+
128
+ // If there is no 'error' event listener then throw.
129
+ if (doError) {
130
+ var er;
131
+ if (args.length > 0)
132
+ er = args[0];
133
+ if (er instanceof Error) {
134
+ // Note: The comments on the `throw` lines are intentional, they show
135
+ // up in Node's output if this results in an unhandled exception.
136
+ throw er; // Unhandled 'error' event
137
+ }
138
+ // At least give some kind of context to the user
139
+ var err = new Error('Unhandled error.' + (er ? ' (' + er.message + ')' : ''));
140
+ err.context = er;
141
+ throw err; // Unhandled 'error' event
142
+ }
143
+
144
+ var handler = events[type];
145
+
146
+ if (handler === undefined)
147
+ return false;
148
+
149
+ if (typeof handler === 'function') {
150
+ ReflectApply(handler, this, args);
151
+ } else {
152
+ var len = handler.length;
153
+ var listeners = arrayClone(handler, len); // eslint-disable-next-line no-redeclare
154
+ for (var i = 0; i < len; ++i)
155
+ ReflectApply(listeners[i], this, args);
156
+ }
157
+
158
+ return true;
159
+ };
160
+
161
+ function _addListener(target, type, listener, prepend) {
162
+ var m;
163
+ var events;
164
+ var existing;
165
+
166
+ checkListener(listener);
167
+
168
+ events = target._events;
169
+ if (events === undefined) {
170
+ events = target._events = Object.create(null);
171
+ target._eventsCount = 0;
172
+ } else {
173
+ // To avoid recursion in the case that type === "newListener"! Before
174
+ // adding it to the listeners, first emit "newListener".
175
+ if (events.newListener !== undefined) {
176
+ target.emit('newListener', type, listener.listener ? listener.listener : listener);
177
+
178
+ // Re-assign `events` because a newListener handler could have caused the
179
+ // this._events to be assigned to a new object
180
+ events = target._events;
181
+ }
182
+ existing = events[type];
183
+ }
184
+
185
+ if (existing === undefined) {
186
+ // Optimize the case of one listener. Don't need the extra array object.
187
+ existing = events[type] = listener;
188
+ ++target._eventsCount;
189
+ } else {
190
+ if (typeof existing === 'function') {
191
+ // Adding the second element, need to change to array.
192
+ existing = events[type] =
193
+ prepend ? [listener, existing] : [existing, listener];
194
+ // If we've already got an array, just append.
195
+ } else if (prepend) {
196
+ existing.unshift(listener);
197
+ } else {
198
+ existing.push(listener);
199
+ }
200
+
201
+ // Check for listener leak
202
+ m = _getMaxListeners(target);
203
+ if (m > 0 && existing.length > m && !existing.warned) {
204
+ existing.warned = true;
205
+ // No error code for this since it is a Warning
206
+ // eslint-disable-next-line no-restricted-syntax
207
+ var w = new Error('Possible EventEmitter memory leak detected. ' +
208
+ existing.length + ' ' + String(type) + ' listeners ' +
209
+ 'added. Use emitter.setMaxListeners() to ' +
210
+ 'increase limit');
211
+ w.name = 'MaxListenersExceededWarning';
212
+ w.emitter = target;
213
+ w.type = type;
214
+ w.count = existing.length;
215
+ ProcessEmitWarning(w);
216
+ }
217
+ }
218
+
219
+ return target;
220
+ }
221
+
222
+ EventEmitter.prototype.addListener = function addListener(type, listener) {
223
+ return _addListener(this, type, listener, false);
224
+ };
225
+
226
+ EventEmitter.prototype.on = EventEmitter.prototype.addListener;
227
+
228
+ EventEmitter.prototype.prependListener =
229
+ function prependListener(type, listener) {
230
+ return _addListener(this, type, listener, true);
231
+ };
232
+
233
+ function onceWrapper() {
234
+ if (!this.fired) {
235
+ this.target.removeListener(this.type, this.wrapFn);
236
+ this.fired = true;
237
+ if (arguments.length === 0)
238
+ return this.listener.call(this.target);
239
+ return this.listener.apply(this.target, arguments);
240
+ }
241
+ }
242
+
243
+ function _onceWrap(target, type, listener) {
244
+ var state = { fired: false, wrapFn: undefined, target: target, type: type, listener: listener };
245
+ var wrapped = onceWrapper.bind(state);
246
+ wrapped.listener = listener;
247
+ state.wrapFn = wrapped;
248
+ return wrapped;
249
+ }
250
+
251
+ EventEmitter.prototype.once = function once(type, listener) {
252
+ checkListener(listener);
253
+ this.on(type, _onceWrap(this, type, listener));
254
+ return this;
255
+ };
256
+
257
+ EventEmitter.prototype.prependOnceListener =
258
+ function prependOnceListener(type, listener) {
259
+ checkListener(listener);
260
+ this.prependListener(type, _onceWrap(this, type, listener));
261
+ return this;
262
+ };
263
+
264
+ // Emits a 'removeListener' event if and only if the listener was removed.
265
+ EventEmitter.prototype.removeListener =
266
+ function removeListener(type, listener) {
267
+ var list, events, position, i, originalListener;
268
+
269
+ checkListener(listener);
270
+
271
+ events = this._events;
272
+ if (events === undefined)
273
+ return this;
274
+
275
+ list = events[type];
276
+ if (list === undefined)
277
+ return this;
278
+
279
+ if (list === listener || list.listener === listener) {
280
+ if (--this._eventsCount === 0)
281
+ this._events = Object.create(null);
282
+ else {
283
+ delete events[type];
284
+ if (events.removeListener)
285
+ this.emit('removeListener', type, list.listener || listener);
286
+ }
287
+ } else if (typeof list !== 'function') {
288
+ position = -1;
289
+
290
+ for (i = list.length - 1; i >= 0; i--) {
291
+ if (list[i] === listener || list[i].listener === listener) {
292
+ originalListener = list[i].listener;
293
+ position = i;
294
+ break;
295
+ }
296
+ }
297
+
298
+ if (position < 0)
299
+ return this;
300
+
301
+ if (position === 0)
302
+ list.shift();
303
+ else {
304
+ spliceOne(list, position);
305
+ }
306
+
307
+ if (list.length === 1)
308
+ events[type] = list[0];
309
+
310
+ if (events.removeListener !== undefined)
311
+ this.emit('removeListener', type, originalListener || listener);
312
+ }
313
+
314
+ return this;
315
+ };
316
+
317
+ EventEmitter.prototype.off = EventEmitter.prototype.removeListener;
318
+
319
+ EventEmitter.prototype.removeAllListeners =
320
+ function removeAllListeners(type) {
321
+ var listeners, events, i;
322
+
323
+ events = this._events;
324
+ if (events === undefined)
325
+ return this;
326
+
327
+ // not listening for removeListener, no need to emit
328
+ if (events.removeListener === undefined) {
329
+ if (arguments.length === 0) {
330
+ this._events = Object.create(null);
331
+ this._eventsCount = 0;
332
+ } else if (events[type] !== undefined) {
333
+ if (--this._eventsCount === 0)
334
+ this._events = Object.create(null);
335
+ else
336
+ delete events[type];
337
+ }
338
+ return this;
339
+ }
340
+
341
+ // emit removeListener for all listeners on all events
342
+ if (arguments.length === 0) {
343
+ var keys = Object.keys(events);
344
+ var key;
345
+ for (i = 0; i < keys.length; ++i) {
346
+ key = keys[i];
347
+ if (key === 'removeListener') continue;
348
+ this.removeAllListeners(key);
349
+ }
350
+ this.removeAllListeners('removeListener');
351
+ this._events = Object.create(null);
352
+ this._eventsCount = 0;
353
+ return this;
354
+ }
355
+
356
+ listeners = events[type];
357
+
358
+ if (typeof listeners === 'function') {
359
+ this.removeListener(type, listeners);
360
+ } else if (listeners !== undefined) {
361
+ // LIFO order
362
+ for (i = listeners.length - 1; i >= 0; i--) {
363
+ this.removeListener(type, listeners[i]);
364
+ }
365
+ }
366
+
367
+ return this;
368
+ };
369
+
370
+ function _listeners(target, type, unwrap) {
371
+ var events = target._events;
372
+
373
+ if (events === undefined)
374
+ return [];
375
+
376
+ var evlistener = events[type];
377
+ if (evlistener === undefined)
378
+ return [];
379
+
380
+ if (typeof evlistener === 'function')
381
+ return unwrap ? [evlistener.listener || evlistener] : [evlistener];
382
+
383
+ return unwrap ?
384
+ unwrapListeners(evlistener) : arrayClone(evlistener, evlistener.length);
385
+ }
386
+
387
+ EventEmitter.prototype.listeners = function listeners(type) {
388
+ return _listeners(this, type, true);
389
+ };
390
+
391
+ EventEmitter.prototype.rawListeners = function rawListeners(type) {
392
+ return _listeners(this, type, false);
393
+ };
394
+
395
+ EventEmitter.listenerCount = function(emitter, type) {
396
+ if (typeof emitter.listenerCount === 'function') {
397
+ return emitter.listenerCount(type);
398
+ } else {
399
+ return listenerCount.call(emitter, type);
400
+ }
401
+ };
402
+
403
+ EventEmitter.prototype.listenerCount = listenerCount;
404
+ function listenerCount(type) {
405
+ var events = this._events;
406
+
407
+ if (events !== undefined) {
408
+ var evlistener = events[type];
409
+
410
+ if (typeof evlistener === 'function') {
411
+ return 1;
412
+ } else if (evlistener !== undefined) {
413
+ return evlistener.length;
414
+ }
415
+ }
416
+
417
+ return 0;
418
+ }
419
+
420
+ EventEmitter.prototype.eventNames = function eventNames() {
421
+ return this._eventsCount > 0 ? ReflectOwnKeys(this._events) : [];
422
+ };
423
+
424
+ function arrayClone(arr, n) {
425
+ var copy = new Array(n);
426
+ for (var i = 0; i < n; ++i)
427
+ copy[i] = arr[i];
428
+ return copy;
429
+ }
430
+
431
+ function spliceOne(list, index) {
432
+ for (; index + 1 < list.length; index++)
433
+ list[index] = list[index + 1];
434
+ list.pop();
435
+ }
436
+
437
+ function unwrapListeners(arr) {
438
+ var ret = new Array(arr.length);
439
+ for (var i = 0; i < ret.length; ++i) {
440
+ ret[i] = arr[i].listener || arr[i];
441
+ }
442
+ return ret;
443
+ }