/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ var random = { // eslint-disable-line no-unused-vars twister: null, /** * Must be called before any other methods can be called to initialize MersenneTwister. * @param {number|null|undefined} seed Value to initialize MersenneTwister. */ init: function (seed) { if (seed === null || seed === undefined) { seed = new Date().getTime() } this.twister = new MersenneTwister() this.twister.seed(seed) }, number: function (limit) { // Returns an integer in [0, limit). Uniform distribution. if (limit === 0) { return limit } if (limit === null || limit === undefined) { limit = 0xffffffff } let x = (0x100000000 / limit) >>> 0 let y = (x * limit) >>> 0 let r do { r = this.twister.int32() } while (y && r >= y) // eslint-disable-line no-unmodified-loop-condition return (r / x) >>> 0 }, float: function () { // Returns a float in [0, 1). Uniform distribution. return this.twister.real2() }, range: function (start, limit) { // Returns an integer in [start, limit]. Uniform distribution. if (isNaN(start) || isNaN(limit)) { logger.traceback() throw new TypeError('random.range() received a non number type: \'' + start + '\', \'' + limit + '\')') } return this.number(limit - start + 1) + start }, ludOneTo: function (limit) { // Returns a float in [1, limit]. The logarithm has uniform distribution. return Math.exp(this.float() * Math.log(limit)) }, item: function (list) { if (!(list instanceof Array || (list !== undefined && typeof list !== 'string' && list.hasOwnProperty('length')))) { logger.traceback() throw new TypeError('this.item() received a non array type: \'' + list + '\'') } return list[this.number(list.length)] }, key: function (obj) { let list = [] for (let i in obj) { list.push(i) } return this.item(list) }, bool: function () { return this.item([true, false]) }, pick: function (obj) { if (typeof obj === 'function') { return obj() } if (obj instanceof Array) { return this.pick(this.item(obj)) } return obj }, chance: function (limit) { if (limit === null || limit === undefined) { limit = 2 } if (isNaN(limit)) { logger.traceback() throw new TypeError('random.chance() received a non number type: \'' + limit + '\'') } return this.number(limit) === 1 }, choose: function (list, flat) { if (!(list instanceof Array)) { logger.traceback() throw new TypeError('random.choose() received a non-array type: \'' + list + '\'') } let total = 0 for (let i = 0; i < list.length; i++) { total += list[i][0] } let n = this.number(total) for (let i = 0; i < list.length; i++) { if (n < list[i][0]) { if (flat === true) { return list[i][1] } else { return this.pick([list[i][1]]) } } n = n - list[i][0] } if (flat === true) { return list[0][1] } return this.pick([list[0][1]]) }, weighted: function (wa) { // More memory-hungry but hopefully faster than random.choose$flat let a = [] for (let i = 0; i < wa.length; ++i) { for (let j = 0; j < wa[i].w; ++j) { a.push(wa[i].v) } } return a }, use: function (obj) { return this.bool() ? obj : '' }, shuffle: function (arr) { let i = arr.length while (i--) { let p = this.number(i + 1) let t = arr[i] arr[i] = arr[p] arr[p] = t } }, shuffled: function (arr) { let newArray = arr.slice() this.shuffle(newArray) return newArray }, subset: function (list, limit) { if (!(list instanceof Array)) { logger.traceback() throw new TypeError('random.some() received a non-array type: \'' + list + '\'') } if (typeof limit !== 'number') { limit = this.number(list.length + 1) } let result = [] for (let i = 0; i < limit; i++) { result.push(this.pick(list)) } return result }, pop: function (arr) { // Removes and returns a random item from an array let i, obj i = this.number(arr.length) obj = arr[i] arr.splice(i, 1) return obj }, hex: function (len) { return Math.floor(Math.random() * ('0x1' + Array(len + 1).join('0'))).toString(16) } } /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ /* * JavaScript version of Mersenne Twister * * @author Yasuharu Okada * */ function MersenneTwister () { // eslint-disable-line no-unused-vars const N = 624 const M = 397 const UPPER_MASK = 0x80000000 const LOWER_MASK = 0x7fffffff const MAG01 = new Int32Array([0, 0x9908b0df]) let mt = new Int32Array(N) /* the array for the state vector */ let mti = 625 this.seed = function (s) { mt[0] = s | 0 for (mti = 1; mti < N; mti++) { mt[mti] = Math.imul(1812433253, mt[mti - 1] ^ (mt[mti - 1] >>> 30)) + mti } } this.export_state = function () { return [mt, mti] } this.import_state = function (s) { mt = s[0] mti = s[1] } this.export_mta = function () { return mt } this.import_mta = function (_mta) { mt = _mta } this.export_mti = function () { return mti } this.import_mti = function (_mti) { mti = _mti } this.int32 = function () { let y, kk if (mti >= N) { /* generate N words at one time */ for (kk = 0; kk < N - M; kk++) { y = ((mt[kk] & UPPER_MASK) | (mt[kk + 1] & LOWER_MASK)) mt[kk] = (mt[kk + M] ^ (y >>> 1) ^ MAG01[y & 0x1]) } for (; kk < N - 1; kk++) { y = ((mt[kk] & UPPER_MASK) | (mt[kk + 1] & LOWER_MASK)) mt[kk] = (mt[kk + (M - N)] ^ (y >>> 1) ^ MAG01[y & 0x1]) } y = ((mt[N - 1] & UPPER_MASK) | (mt[0] & LOWER_MASK)) mt[N - 1] = (mt[M - 1] ^ (y >>> 1) ^ MAG01[y & 0x1]) mti = 0 } y = mt[mti++] /* Tempering */ y = y ^ (y >>> 11) y = y ^ ((y << 7) & 0x9d2c5680) y = y ^ ((y << 15) & 0xefc60000) y = y ^ (y >>> 18) return y >>> 0 } this.real2 = function () { return ((this.int32() >>> 5) * 67108864.0 + (this.int32() >>> 6)) / 9007199254740992.0 } } var utils = {} // eslint-disable-line no-unused-vars /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ utils.platform = { platform: function () { let platform = {} const userAgent = (navigator.userAgent).toLowerCase() const language = navigator.language || navigator.browserLanguage const version = platform.version = (userAgent.match(/.*(?:rv|chrome|webkit|opera|ie)[/: ](.+?)([ );]|$)/) || [])[1] const webkitVersion = (userAgent.match(/webkit\/(.+?) /) || [])[1] platform.windows = platform.isWindows = !!/windows/.test(userAgent) platform.mac = platform.isMac = !!/macintosh/.test(userAgent) || (/mac os x/.test(userAgent) && !/like mac os x/.test(userAgent)) platform.lion = platform.isLion = !!(/mac os x 10_7/.test(userAgent) && !/like mac os x 10_7/.test(userAgent)) platform.iPhone = platform.isiPhone = !!/iphone/.test(userAgent) platform.iPod = platform.isiPod = !!/ipod/.test(userAgent) platform.iPad = platform.isiPad = !!/ipad/.test(userAgent) platform.iOS = platform.isiOS = platform.iPhone || platform.iPod || platform.iPad platform.android = platform.isAndroid = !!/android/.test(userAgent) platform.opera = /opera/.test(userAgent) ? version : 0 platform.isOpera = !!platform.opera platform.msie = /msie/.test(userAgent) && !platform.opera ? version : 0 platform.isIE = !!platform.msie platform.isIE8OrLower = !!(platform.msie && parseInt(platform.msie, 10) <= 8) platform.mozilla = /mozilla/.test(userAgent) && !/(compatible|webkit|msie)/.test(userAgent) ? version : 0 platform.isMozilla = !!platform.mozilla platform.webkit = /webkit/.test(userAgent) ? webkitVersion : 0 platform.isWebkit = !!platform.webkit platform.chrome = /chrome/.test(userAgent) ? version : 0 platform.isChrome = !!platform.chrome platform.mobileSafari = /apple.*mobile/.test(userAgent) && platform.iOS ? webkitVersion : 0 platform.isMobileSafari = !!platform.mobileSafari platform.iPadSafari = platform.iPad && platform.isMobileSafari ? webkitVersion : 0 platform.isiPadSafari = !!platform.iPadSafari platform.iPhoneSafari = platform.iPhone && platform.isMobileSafari ? webkitVersion : 0 platform.isiPhoneSafari = !!platform.iphoneSafari platform.iPodSafari = platform.iPod && platform.isMobileSafari ? webkitVersion : 0 platform.isiPodSafari = !!platform.iPodSafari platform.isiOSHomeScreen = platform.isMobileSafari && !/apple.*mobile.*safari/.test(userAgent) platform.safari = platform.webkit && !platform.chrome && !platform.iOS && !platform.android ? webkitVersion : 0 platform.isSafari = !!platform.safari platform.language = language.split('-', 1)[0] platform.current = platform.msie ? 'msie' : platform.mozilla ? 'mozilla' : platform.chrome ? 'chrome' : platform.safari ? 'safari' : platform.opera ? 'opera' : platform.mobileSafari ? 'mobile-safari' : platform.android ? 'android' : 'unknown' function platformName (candidates) { for (let i = 0; i < candidates.length; i++) { if (candidates[i] in window) { return 'window.' + candidates[i] } if (candidates[i] in navigator) { return 'navigator.' + candidates[i] } } return undefined } platform.GUM = platformName(['getUserMedia', 'webkitGetUserMedia', 'mozGetUserMedia', 'msGetUserMedia', 'getGUM']) platform.PeerConnection = platformName(['webkitRTCPeerConnection', 'mozRTCPeerConnection', 'msPeerConnection']) platform.IceCandidate = platformName(['mozRTCIceCandidate', 'RTCIceCandidate']) platform.SessionDescription = platformName(['mozRTCSessionDescription', 'RTCSessionDescription']) platform.URL = platformName(['URL', 'webkitURL']) platform.AudioContext = platformName(['AudioContext', 'webkitAudioContext']) platform.OfflineAudioContext = platformName(['OfflineAudioContext', 'webkitOfflineAudioContext']) platform.MediaSource = platformName(['MediaSource', 'WebKitMediaSource']) platform.SpeechRecognition = platformName(['SpeechRecognition', 'webkitSpeechRecognition']) platform.SpeechGrammarList = platformName(['SpeechGrammarList', 'webkitSpeechGrammarList']) /* function findWebGLContextName (candidates) { var canvas = document.createElement('canvas') for (var i = 0; i < candidates.length; i++) { var name = candidates[i] try { if (canvas.getContext(name)) { return name } } catch (e) {} } return null } */ platform.WebGL = 'webgl' // findWebGLContextName(["webgl", "experimental-webgl", "webkit-3d"]); platform.WebGL2 = 'webgl2' // findWebGLContextName(["webgl2", "experimental-webgl2"]); platform.captureStreamUntilEnded = 'captureStreamUntilEnded' if (platform.isMozilla) { platform.captureStreamUntilEnded = 'mozCaptureStreamUntilEnded' } platform.srcObject = 'srcObject' if (platform.isMozilla) { platform.srcObject = 'mozSrcObject' } return platform } } /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ var websocket = null var logger = (function () { // eslint-disable-line no-unused-vars const sep = '\n/* ### NEXT TESTCASE ############################## */' const color = { red: '\u0033[1;31m', green: '\u0033[1;32m', clear: '\u0033[0m' } if (utils.platform.isWindows) { color.red = '' color.green = '' color.clear = '' } function console (msg) { if (websocket) { websocket.send(msg) } if (typeof window === 'undefined') { print(msg) } else if (window.dump) { window.dump(msg) } else if (window.console && window.console.log) { window.console.log(msg) } else { throw new Error('Unable to run console logger.') } } function dump (msg) { console(msg) } function testcase (msg) { dump('/*L*/ ' + JSON.stringify(msg) + '\n') } function dumpln (msg) { dump(msg + '\n') } function error (msg) { dumpln(color.red + msg + color.clear) } function JSError (msg) { error(comment(msg)) } function comment (msg) { return '/* ' + msg + ' */' } function separator () { dumpln(color.green + sep + color.clear) } function traceback () { error('===[ Traceback ]') try { throw new Error() } catch (e) { dump(e.stack || e.stacktrace || '') } error('===') } return { console: console, dump: dump, error: error, JSError: JSError, dumpln: dumpln, comment: comment, testcase: testcase, separator: separator, traceback: traceback } })() var make = {} // eslint-disable-line no-unused-vars /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ make.shaders = { fragment1: function () { return random.pick([ [ '#ifdef GL_ES', 'precision mediump float;', '#endif', 'varying vec4 vColor;', 'void main() {', 'gl_FragColor=vColor;', '}' ], [ 'varying highp vec2 vTextureCoord;', 'varying highp vec3 vLighting;', 'uniform sampler2D uSampler;', 'void main(void) {', 'highp vec4 texelColor = texture2D(uSampler, vec2(vTextureCoord.s, vTextureCoord.t));', 'gl_FragColor = vec4(texelColor.rgb * vLighting, texelColor.a);', '}' ] ]) }, vertex1: function () { return random.pick([ [ 'attribute vec4 aVertex;', 'attribute vec4 aColor;', 'varying vec4 vColor;', 'void main(){', 'vColor=aColor;', 'gl_Position=aVertex;', '}' ], [ 'attribute highp vec3 aVertexNormal;', 'attribute highp vec3 aVertexPosition;', 'attribute highp vec2 aTextureCoord;', 'uniform highp mat4 uNormalMatrix;', 'uniform highp mat4 uMVMatrix;', 'uniform highp mat4 uPMatrix;', 'varying highp vec2 vTextureCoord;', 'varying highp vec3 vLighting;', 'void main(void) {', 'gl_Position = uPMatrix * uMVMatrix * vec4(aVertexPosition, 1.0);', 'vTextureCoord = aTextureCoord;', 'highp vec3 ambientLight = vec3(0.6, 0.6, 0.6);', 'highp vec3 directionalLightColor = vec3(0.5, 0.5, 0.75);', 'highp vec3 directionalVector = vec3(0.85, 0.8, 0.75);', 'highp vec4 transformedNormal = uNormalMatrix * vec4(aVertexNormal, 1.0);', 'highp float directional = max(dot(transformedNormal.xyz, directionalVector), 0.0);', 'vLighting = ambientLight + (directionalLightColor * directional);', '}' ] ]) }, fragment2: function () { return random.pick([ [ 'varying highp vec2 vTextureCoord;', 'varying highp vec3 vLighting;', 'uniform sampler2D uSampler;', 'void main(void) {', 'highp vec4 texelColor = texture2D(uSampler, vec2(vTextureCoord.s, vTextureCoord.t));', 'gl_FragColor = vec4(texelColor.rgb * vLighting, texelColor.a);', '}' ], [ '#version proto-200', 'uniform sampler2D albedoMap;', 'uniform sampler2D normalMap;', 'varying vec3 varyingTangent;', 'varying vec3 varyingBitangent;', 'varying vec3 varyingNormal;', 'varying vec2 varyingUV;', 'void main(void) {', 'vec3 albedo=texture2D(albedoMap,varyingUV).rgb;', 'vec3 normal=texture2D(normalMap,varyingUV).rgb*2.0-1.0;', 'float specularFactor=pow((albedo.r+albedo.g+albedo.b)*0.33,2.0);', 'float specularHardness=2.0;', 'vec3 spaceNormal=varyingTangent*normal.x+varyingBitangent*normal.y+varyingNormal*normal.z;', 'gl_FragData[0]=vec4(albedo,1.0);', 'gl_FragData[1]=vec4(spaceNormal*0.5 +0.5,1.0);', 'gl_FragData[2]=vec4(specularFactor,specularHardness*0.1,0.0,1.0);', '}' ] ]) }, vertex2: function () { return random.pick([ [ 'attribute highp vec3 aVertexNormal;', 'attribute highp vec3 aVertexPosition;', 'attribute highp vec2 aTextureCoord;', 'uniform highp mat4 uNormalMatrix;', 'uniform highp mat4 uMVMatrix;', 'uniform highp mat4 uPMatrix;', 'varying highp vec2 vTextureCoord;', 'varying highp vec3 vLighting;', 'void main(void) {', 'gl_Position = uPMatrix * uMVMatrix * vec4(aVertexPosition, 1.0);', 'vTextureCoord = aTextureCoord;', 'highp vec3 ambientLight = vec3(0.6, 0.6, 0.6);', 'highp vec3 directionalLightColor = vec3(0.5, 0.5, 0.75);', 'highp vec3 directionalVector = vec3(0.85, 0.8, 0.75);', 'highp vec4 transformedNormal = uNormalMatrix * vec4(aVertexNormal, 1.0);', 'highp float directional = max(dot(transformedNormal.xyz, directionalVector), 0.0);', 'vLighting = ambientLight + (directionalLightColor * directional);', '}' ], [ '#version proto-200', 'attribute vec3 vertexPosition;', 'attribute vec3 vertexTangent;', 'attribute vec3 vertexBitangent;', 'attribute vec3 vertexNormal;', 'attribute vec2 vertexUV;', 'uniform mat4 modelMatrix;', 'uniform mat4 viewMatrix;', 'varying vec3 varyingTangent;', 'varying vec3 varyingBitangent;', 'varying vec3 varyingNormal;', 'varying vec2 varyingUV;', 'void main(void){', 'gl_Position=viewMatrix*(modelMatrix*vec4(vertexPosition,1.0));', 'gl_Position.xy=gl_Position.xy*0.5+(float(gl_InstanceID)-0.5);', 'varyingTangent=(modelMatrix*vec4(vertexTangent,0.0)).xyz;', 'varyingBitangent=(modelMatrix*vec4(vertexBitangent,0.0)).xyz;', 'varyingNormal=(modelMatrix*vec4(vertexNormal,0.0)).xyz;', 'varyingUV = vertexUV;', '}' ] ]) }, shaderPair: function (v, f) { let i = random.number(v.length) return { vertex: utils.common.quote(v[i].join('\n')), fragment: utils.common.quote(f[i].join('\n')) } } } /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ make.colors = { any: function () { return random.pick([ make.colors.rgb, make.colors.hsl, make.colors.keyword ]) }, rgb: function () { let values switch (random.number(4)) { case 0: // Rgb functional notation if (random.bool()) { // Ints values = [random.number(255), random.number(255), random.number(255)] } else { // Percents values = ['%' + random.number(255), '%' + random.number(255), '%' + random.number(255)] } return 'rgba(' + values.join(',') + ')' case 1: // Rgba functional notation values = [random.number(255), random.number(255), random.number(255), random.float()] return 'rgba(' + values.join(',') + ')' case 2: // 4 char hex return '#' + random.hex(4) default: // 8 char hex return '#' + random.hex(8) } }, hsl: function () { let values, opt switch (random.number(4)) { case 0: values = [random.number(255), '%' + random.number(255), '%' + random.number(255)] return 'hsl(' + values.join(',') + ')' case 1: values = [random.number(255), '%' + random.number(255), '%' + random.number(255), '%' + random.number(255)] return 'hsl(' + values.join(',') + ')' case 2: opt = random.pick(['deg', 'rad', 'grad', 'turn']) values = [random.number(255) + opt, '%' + random.number(255), '%' + random.number(255), '%' + random.number(255)] return 'hsl(' + values.join(',') + ')' default: values = [random.number(255), '%' + random.number(255), '%' + random.number(255), random.float()] return 'hsl(' + values.join(',') + ')' } }, keyword: function () { return random.pick([ 'lime', 'red', 'blue', 'invert', 'currentColor', 'ActiveBorder', 'ActiveCaption', 'AppWorkspace', 'Background', 'ButtonFace', 'ButtonHighlight', 'ButtonShadow', 'ButtonText', 'CaptionText', 'GrayText', 'Highlight', 'HighlightText', 'InactiveBorder', 'InactiveCaption', 'InactiveCaptionText', 'InfoBackground', 'InfoText', 'Menu', 'MenuText', 'Scrollbar', 'ThreeDDarkShadow', 'ThreeDFace', 'ThreeDHighlight', 'ThreeDLightShadow', 'ThreeDShadow', 'Window', 'WindowFrame', 'WindowText', '-moz-ButtonDefault', '-moz-ButtonHoverFace', '-moz-ButtonHoverText', '-moz-CellHighlight', '-moz-CellHighlightText', '-moz-Combobox', '-moz-ComboboxText', '-moz-Dialog', '-moz-DialogText', '-moz-dragtargetzone', '-moz-EvenTreeRow', '-moz-Field', '-moz-FieldText', '-moz-html-CellHighlight', '-moz-html-CellHighlightText', '-moz-mac-accentdarkestshadow', '-moz-mac-accentdarkshadow', '-moz-mac-accentface', '-moz-mac-accentlightesthighlight', '-moz-mac-accentlightshadow', '-moz-mac-accentregularhighlight', '-moz-mac-accentregularshadow', '-moz-mac-chrome-active', '-moz-mac-chrome-inactive', '-moz-mac-focusring', '-moz-mac-menuselect', '-moz-mac-menushadow', '-moz-mac-menutextselect', '-moz-MenuHover', '-moz-MenuHoverText', '-moz-MenuBarText', '-moz-MenuBarHoverText', '-moz-nativehyperlinktext', '-moz-OddTreeRow', '-moz-win-communicationstext', '-moz-win-mediatext', '-moz-activehyperlinktext', '-moz-default-background-color', '-moz-default-color', '-moz-hyperlinktext', '-moz-visitedhyperlinktext' ]) } } /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ make.mime = { any: function () { return random.pick([ make.mime.standard, make.mime.xml, make.mime.image, make.mime.media, make.mime.form ]) }, standard: function () { return random.pick([ 'text/html', 'text/html; charset=utf-8', 'text/plain', 'text/css', 'text/javascript', 'foo/bar', 'application/octet-stream', 'application/x-shockwave-flash', 'application/x-test' ]) }, xml: function () { return random.pick([ 'application/xml', 'text/xml', 'application/xhtml+xml', 'image/svg+xml', 'application/vnd.mozilla.xul+xml', 'application/rss+xml', 'application/rdf+xml', 'application/xslt+xml' ]) }, image: function () { return random.pick([ 'image/jpeg', 'image/gif', 'image/png', 'image/mng', 'image/*' ]) }, media: function () { return random.pick([ 'audio/mpeg', 'audio/ogg', 'audio/ogg; codecs=vorbis', 'video/ogg', 'video/ogg; codecs="theora, vorbis"', 'video/mp4', 'video/mp4; codecs="avc1.42E01E, mp4a.40.2"' ]) }, form: function () { return random.pick([ 'application/x-www-form-urlencoded', 'multipart/form-data', 'text/plain' ]) } } /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ make.strings = { toString: function (object) { return object ? object.toSource() : '' + object }, string: function (maxlen) { let s = '' if (maxlen === null || maxlen === undefined) { maxlen = make.number.rangeNumber() } for (let i = 0; i < maxlen; i++) { // Todo: s += String.fromCodePoint(Random.pick(make.fonts.layoutCharCodes)); s += 'A' } return s }, quotedString: function (maxlen) { return utils.common.quote(make.strings.string(maxlen)) }, stringFromBlocks: function (set, maxlen) { let s = '' for (let i = 0; i < random.number(maxlen || 255); i++) { s += random.pick(set) } return s }, digitsHex: function (n) { let s = '' while (n-- > 0) { s += (random.number(16)).toString(16) } return s } } /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ make.arrays = { filledArray: function (fn, limit) { let array = [] let size = limit || random.number(make.number.tinyNumber) for (let i = 0; i < size; i++) { array.push(fn()) } return array } } /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ make.font = { globalValue: function () { return random.pick(['inherit', 'initial', 'unset']) }, style: function () { return random.pick(['italic', 'normal', 'oblique', 'inherit']) }, variant: function () { return random.pick(['normal', 'small-caps', 'inherit']) }, weight: function () { return random.pick([ /* standard */ ['normal', 'bold'], /* Relative to the parent */ ['bolder', 'lighter'], /* numeric values */ [100, 200, 300, 400, 500, 600, 700, 800, 900] ]) }, size: function () { return random.pick([ /* values */ ['xx-small', 'x-small', 'small', 'medium', 'large', 'x-large', 'xx-large'], /* values */ ['larger', 'smaller'], /* values */ make.number.unsignedNumber() + make.unit.unit(), /* values */ make.unit.percent() ]) }, genericFamily: function () { return random.pick(['serif', 'sans-serif', 'cursive', 'fantasy', 'monospace']) }, familyName: function () { return random.pick(['Times New Roman', 'Arial', 'Courier', 'Helvetica']) }, family: function () { let s = random.pick(make.font.familyName) if (random.chance(8)) { s += ', ' + random.pick(make.font.genericFamily) } return s }, font: function () { let s = '' if (random.chance(4)) { s += random.pick(make.font.style) + ' ' } if (random.chance(4)) { s += random.pick(make.font.variant) + ' ' } if (random.chance(4)) { s += random.pick(make.font.weight) + ' ' } if (random.chance(4)) { s += make.number.any() + '/' } s += make.font.size() s += ' ' s += make.font.family() return '\'' + s + '\'' } } /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ make.unit = { unit: function () { return random.pick([ 'px', 'em', 'ex', 'ch', 'rem', 'mm', 'cm', 'in', 'pt', 'pc', '%' ]) }, length: function () { return make.number.any() + make.unit.unit() }, percent: function () { return make.number.any() + '%' } } /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ make.text = { lineEnd: function () { return random.pick([ '\n', '\r', '\r\n', '\n\r' ]) }, controlChar: function () { return random.pick([ '\b', '\t', '\n', '\v', '\f', '\r', '\0', '\c', '\a', '\e' // eslint-disable-line no-useless-escape ]) }, token: function () { return random.pick([ '*', '+', '%', '-', '!', '^', ':', '|', '&', '<', '>', '.', '"', '#', ' ', ';', ',', '{', '}', '(', ')', '[', ']', '/', '\\', '/*', '*/' ]) }, charset: function () { return random.pick([ 'UTF-8', 'ISO-8859-1' ]) }, language: function () { // https://gist.github.com/tonyhb/635401 return random.pick([ 'en-US', 'en', 'de' ]) }, layoutCharCodes: function () { return random.pick([ 0, // null 160, // non-breaking space 0x005C, // backslash, but in some countries, represents local currency symbol (e.g. yen) 0x00AD, // soft hyphen 0x0BCC, // a Tamil character that is displayed as three glyphs // http://unicode.org/charts/PDF/U2000.pdf 0x200B, // zero-width space 0x200C, // zero-width non-joiner 0x200D, // zero-width joiner 0x200E, // left-to-right mark 0x200F, // right-to-left mark 0x2011, // non-breaking hyphen 0x2027, // hyphenation point 0x2028, // line separator 0x2029, // paragraph separator 0x202A, // left-to-right embedding 0x202B, // right-to-left embedding 0x202C, // pop directional formatting 0x202D, // left-to-right override 0x202E, // right-to-left override 0x202F, // narrow no-break space 0x2060, // word joiner 0x2061, // function application (one of several invisible mathematical operators) // http://unicode.org/charts/PDF/U3000.pdf 0x3000, // ideographic space (CJK) // http://unicode.org/charts/PDF/U0300.pdf 0x0301, // combining acute accent (if it appears after "a", it turns into "a" with an accent) // Arabic has the interesting property that most letters connect to the next letter. // Some code calls this "shaping". 0x0643, // arabic letter kaf 0x0645, // arabic letter meem 0x06CD, // arabic letter yeh with tail 0xFDDE, // invalid unicode? but somehow associated with arabic. // http://unicode.org/reports/tr36/tr36-7.html#Buffer_Overflows // Characters with especially high expansion factors when they go through various unicode "normalizations" 0x1F82, 0xFDFA, 0xFB2C, 0x0390, // 0x1D160, // hmm, need surrogates // Characters with especially high expansion factors when lowercased or uppercased 0x023A, 0x0041, 0xDC1D, // a low surrogate 0xDB00, // a high surrogate // UFFF0.pdf 0xFFF9, // interlinear annotation anchor 0xFFFA, // interlinear annotation seperator 0xFFFB, // interlinear annotation terminator 0xFFFC, // object replacement character 0xFFFD, // replacement character 0xFEFF, // zero width no-break space 0xFFFF, // not a character 0x00A0, // no-break space 0x2426, 0x003F, 0x00BF, 0xDC80, 0xDCFF, // http://en.wikipedia.org/wiki/Mapping_of_Unicode_characters 0x205F, // mathematical space 0x2061, // mathematical function application 0x2064, // mathematical invisible separator 0x2044 // fraction slash character ]) }, // http://www.unicode.org/Public/6.0.0/ucd/UnicodeData.txt unicodeCombiningCharacters: function () { return random.item([ [0x0300, 0x036F], // Combining Diacritical Marks [0x0483, 0x0489], [0x07EB, 0x07F3], [0x135D, 0x135F], [0x1A7F, 0x1A7F], [0x1B6B, 0x1B73], [0x1DC0, 0x1DFF], // Combining Diacritical Marks Supplement [0x20D0, 0x2DFF], [0x3099, 0x309A], [0xA66F, 0xA6F1], [0xA8E0, 0xA8F1], [0xFE20, 0xFE26], // Combining Half Marks [0x101FD, 0x101FD], [0x1D165, 0x1D169], [0x1D16D, 0x1D172], [0x1D17B, 0x1D18B], [0x1D1AA, 0x1D1AD], [0x1D242, 0x1D244] ]) }, unicodeBMP: function () { return random.item([ // BMP = Basic Multilingual Plane [0x0000, 0xFFFF] ]) }, unicodeSMP: function () { return random.item([ // SMP = Supplementary Multilingual Plane [0x10000, 0x13FFF], [0x16000, 0x16FFF], [0x1B000, 0x1BFFF], [0x1D000, 0x1DFFF], [0x1F000, 0x1FFFF] ]) }, unicodeSIP: function () { return random.item([ // SIP = Supplementary Ideographic Plane [0x20000, 0x2BFFF], [0x2F000, 0x2FFFF] ]) }, unicodeSSP: function () { return random.item([ // SSP = Supplementary Special-purpose Plane [0xE0000, 0xE0FFF] ]) }, registeredFontFeatures: function () { return random.pick([ 'aalt', 'abvf', 'abvm', 'abvs', 'afrc', 'akhn', 'blwf', 'blwm', 'blws', 'calt', 'case', 'ccmp', 'cfar', 'cjct', 'clig', 'cpct', 'cpsp', 'cswh', 'curs', 'cv01-cv99', 'c2pc', 'c2sc', 'dist', 'dlig', 'dnom', 'expt', 'falt', 'fin2', 'fin3', 'fina', 'frac', 'fwid', 'half', 'haln', 'halt', 'hist', 'hkna', 'hlig', 'hngl', 'hojo', 'hwid', 'init', 'isol', 'ital', 'jalt', 'jp78', 'jp83', 'jp90', 'jp04', 'kern', 'lfbd', 'liga', 'ljmo', 'lnum', 'locl', 'ltra', 'ltrm', 'mark', 'med2', 'medi', 'mgrk', 'mkmk', 'mset', 'nalt', 'nlck', 'nukt', 'numr', 'onum', 'opbd', 'ordn', 'ornm', 'palt', 'pcap', 'pkna', 'pnum', 'pref', 'pres', 'pstf', 'psts', 'pwid', 'qwid', 'rand', 'rkrf', 'rlig', 'rphf', 'rtbd', 'rtla', 'rtlm', 'ruby', 'salt', 'sinf', 'size', 'smcp', 'smpl', 'ss01', 'ss02', 'ss03', 'ss04', 'ss05', 'ss06', 'ss07', 'ss08', 'ss09', 'ss10', 'ss11', 'ss12', 'ss13', 'ss14', 'ss15', 'ss16', 'ss17', 'ss18', 'ss19', 'ss20', 'subs', 'sups', 'swsh', 'titl', 'tjmo', 'tnam', 'tnum', 'trad', 'twid', 'unic', 'valt', 'vatu', 'vert', 'vhal', 'vjmo', 'vkna', 'vkrn', 'vpal', 'vrt2', 'zero' ]) }, assignmentOperator: function () { return random.pick([ '=', '-=', '+=', '*=', '/=' ]) }, arithmeticOperator: function () { return random.pick([ '%', '-', '+', '*', '/' ]) }, currency: function () { return random.pick([ // https://en.wikipedia.org/wiki/ISO_4217 'USD', 'USS', 'USN', 'EUR', 'CHF', 'GBP', 'XAG', 'XBA', 'XBB', 'XBC', 'XBD', 'XSU', 'XTS', 'XXX' ]) } } /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ make.files = { image: function () { return utils.quote(random.pick([ 'data:image/gif;base64,R0lGODlhAQABAIAAAP///wAAACwAAAAAAQABAAACAkQBADs=', 'data:image/gif;base64,R0lGODlhAQABAIAAAP///wAAACH5BAEAAAAALAAAAAABAAEAAAICRAEAOw==', 'data:image/gif;base64,R0lGODlhAQABAAAAACwAAAAAAQABAAA=', 'data:image/gif;base64,R0lGODlhAQABAAAAACw=', 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAAAAAA6fptVAAAACklEQVQYV2P4DwABAQEAWk1v8QAAAABJRU5ErkJggg==', 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAQAAAC1HAwCAAAAC0lEQVQYV2NgYAAAAAMAAWgmWQ0AAAAASUVORK5CYII=', 'media/images/image1.jpg', 'media/images/image3.jpg' ])) }, video: function () { return utils.quote(random.pick([ 'media/video/video1.webm', 'media/video/video2.webm' ])) }, audio: function () { return utils.quote(random.pick([ 'media/audio/mono-uncompressed-8bit-8000hz.wav', 'media/audio/mono-uncompressed-8bit-44100hz.wav', 'media/audio/mono-uncompressed-32bit-8000hz.wav', 'media/audio/mono-uncompressed-32bit-44100hz.wav' ])) }, webvtt: function () { return utils.quote(random.pick([ // 'data:text/vtt,' + encodeURIComponent('WEBVTT\n\n00:00:00.000 --> 00:00:00.001\ntest');, 'media/video/sample.vtt' ])) }, file: function () { return random.pick([ make.files.image, make.files.video, make.files.audio ]) } } /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ make.network = { sdp: function () { // session description protocol template return [ 'v=0', 'o=Mozilla-SIPUA 23597 0 IN IP4 0.0.0.0', 's=SIP Call', 't=0 0', 'a=ice-ufrag:f5fda439', 'a=ice-pwd:d0df8e2904bdbd29587966e797655970', 'a=fingerprint:sha-256 DF:69:78:20:8D:2E:08:CE:49:82:A3:11:79:1D:BF:B5:49:49:2D:32:82:2F:0D:88:84:A7:C6:63:23:63:A9:0F', 'm=audio 52757 RTP/SAVPF 109 0 8 101', 'c=IN IP4 192.168.129.33', 'a=rtpmap:109 opus/48000/2', 'a=ptime:20', 'a=rtpmap:0 PCMU/8000', 'a=rtpmap:8 PCMA/8000', 'a=rtpmap:101 telephone-event/8000', 'a=fmtp:101 0-15', 'a=sendrecv', 'a=candidate:0 1 UDP 2113601791 192.168.129.33 52757 typ host', 'a=candidate:0 2 UDP 2113601790 192.168.129.33 59738 typ host', 'm=video 63901 RTP/SAVPF 120', 'c=IN IP4 192.168.129.33', 'a=rtpmap:120 VP8/90000', 'a=sendrecv', 'a=candidate:0 1 UDP 2113601791 192.168.129.33 63901 typ host', 'a=candidate:0 2 UDP 2113601790 192.168.129.33 54165 typ host', 'm=application 65080 SCTP/DTLS 5000', 'c=IN IP4 192.168.129.33', 'a=fmtp:5000 protocol=webrtc-datachannel;streams=16', 'a=sendrecv', 'a=candidate:0 1 UDP 2113601791 192.168.129.33 65080 typ host', 'a=candidate:0 2 UDP 2113601790 192.168.129.33 62658 typ host' ].join('\n') }, PeerConnectionProtocols: function () { return ['turn', 'turns', 'stun', 'stuns'] }, randomIPv4: function () { return random.pick([random.number(255), make.number.any]) + '.' + random.pick([random.number(255), make.number.any]) + '.' + random.pick([random.number(255), make.number.any]) + '.' + random.pick([random.number(255), make.number.any]) }, randomIPv6: function () { return '[' + make.strings.stringFromBlocks([':', function () { return make.strings.digitsHex(random.range(1, 4)) }]) + ']' }, goodHostnames: function () { return [ '0.0.0.0', '127.0.0.1:8080' ] }, badHostnames: function () { return [ 'google.org:8080', '::1', '[::192.9.5.5]:42', '2001:db8:85a3::8a2e:370:3478', '2001:db8:85a3:0:0:8a2e:370:3478', '::ffff:192.0.2.1', '0000:0000:0000:0000:0000:0000:0000:0001', '::192.0.2.128', '::ffff:192.0.2.128', '2001:db8::1:2', '2001:db8::1:1:1:1:1' ] }, randomBitmask: function (list) { if (list.length <= 1) { return list.join('') } let max = random.range(2, list.length) let mask = random.pick(list) for (let i = 1; i < max; i++) { mask += '|' + random.pick(list) } return mask } } /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ make.number = { bool: function () { return random.bool() }, float: function () { let n if (random.chance(32)) { switch (random.number(4)) { case 0: n = random.range(Number.MAX_VALUE, Number.MIN_VALUE) break case 1: n = Math.pow(10, 1) / Math.pow(10, random.number(307)) break case 2: n = Math.pow(2, random.float() * random.float() * 64) break case 3: n = Math.pow(10, random.range(1, 9)) / Math.pow(10, random.range(1, 9)) break } return n } switch (random.number(6)) { default: n = random.float() } return n }, rangeNumber: function () { return random.pick([1, 2, 3, 4, 6, 8, 16, 32, 64, make.number.tinyNumber]) }, tinyNumber: function () { return Math.pow(2, random.number(12)) }, unsignedNumber: function () { if (random.chance(2)) { return Math.abs(make.number.any()) } return Math.pow(2, random.number(65)) + random.number(3) - 1 }, evenNumber: function (number) { return number % 2 === 1 ? ++number : number }, anynumber: function () { let value = random.choose([ [10, make.number.float], [10, [make.number.rangeNumber, make.number.tinyNumber]], [1, make.number.unsignedNumber] ]) return random.chance(10) ? -value : value } } /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ make.types = { random: function () { return random.item([ 'true', 'null', '(new Object())', 'undefined', '{}', '[]', '\'\'', 'function() {}' ]) } } /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ utils.script = { methodHead: function (list, numOptional) { if (isNaN(numOptional)) { numOptional = 0 } let arity = list.length - random.number(numOptional) let params = [] for (let i = 0; i < arity; i++) { params.push(random.pick([list[i]])) } return '(' + params.join(', ') + ')' }, methodCall: function (objectName, methodHash) { if (!utils.common.getKeysFromHash(methodHash).length || !objectName) { return '' } let methodName = random.key(methodHash) let methodArgs = methodHash[methodName] if (typeof (methodArgs) === 'function') { // Todo: Hmmmm.. return methodArgs() } return objectName + '.' + methodName + utils.script.methodHead(methodArgs) }, setAttribute: function (objectName, attributeHash) { if (!utils.common.getKeysFromHash(attributeHash).length || !objectName) { return '' } let attributeName = random.key(attributeHash) let attributeValue = random.pick(attributeHash[attributeName]) let operator = ' = ' /* if (typeof(attributeValue) == "number" && Random.chance(8)) { operator = " " + Make.randomAssignmentOperator() + " "; } if (typeof(attributeValue) == "string") { attributeValue = "'" + attributeValue + "'"; } */ return objectName + '.' + attributeName + operator + attributeValue + ';' }, makeConstraint: function (keys, values) { let o = {} let n = random.range(0, keys.length) while (n--) { o[random.pick(keys)] = random.pick(values) } return o }, makeRandomOptions: function (baseObject) { let o = {} let unique = random.some(Object.keys(baseObject)) for (let i = 0; i < unique.length; i++) { o[unique[i]] = random.pick(baseObject[unique[i]]) } return JSON.stringify(o) }, safely: function (s) { if (window.debug) { return 'try { ' + s + ' } catch(e) { logger.JSError(e); }' } return 'try { ' + s + ' } catch(e) { }' }, makeLoop: function (s, max) { return 'for (let i = 0; i < ' + (max || make.number.rangeNumber()) + '; i++) {' + s + '}' }, makeArray: function (type, arrayLength, cb) { if (type === null || type === undefined) { type = random.index(['Uint8', 'Float32']) } switch (random.number(8)) { case 0: let src = 'function() { let buffer = new ' + type + 'Array(' + arrayLength + ');' src += utils.script.makeLoop('buffer[i] = ' + cb() + ';', arrayLength) src += 'return buffer;}()' return src case 1: return 'new ' + type + 'Array([' + make.arrays.filledArray(cb, arrayLength) + '])' default: return 'new ' + type + 'Array(' + arrayLength + ')' } }, randListIndex: function (objName) { return random.number() + ' % ' + o.pick(objName) + '.length' }, addElementToBody: function (name) { return '(document.body || document.documentElement).appendChild' + utils.script.methodHead([name]) }, forceGC: function () { if (utils.platform.isMozilla) { } if (utils.platform.isChrome) { if (window.GCController) { return GCController.collect() // eslint-disable-line no-undef } } if (utils.platform.isSafari) { } if (utils.platform.isIE) { } }, getRandomElement: function () { return 'document.getElementsByTagName(\'*\')[' + random.number(document.getElementsByTagName('*').length) + ']' } } /* eslint no-extend-native: ["error", { "exceptions": ["String", "Array"] }] */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ Object.defineProperty(String.prototype, 'fromCodePoint', function () { let chars = [] let point, offset, units, i for (i = 0; i < arguments.length; ++i) { point = arguments[i] offset = point - 0x10000 units = point > 0xFFFF ? [ 0xD800 + (offset >> 10), 0xDC00 + (offset & 0x3FF)] : [point] chars.push(String.fromCharCode.apply(null, units)) } return chars.join('') }) Object.defineProperty(String.prototype, 'endsWith', function (str) { return this.match(str + '$') === str }) Object.defineProperty(String.prototype, 'startsWith', function (str) { return this.match('^' + str) === str }) Object.defineProperty(String.prototype, 'trim', function () { return this.replace(/^[\s\xA0]+/, '').replace(/[\s\xA0]+$/, '') }) Object.defineProperty(String.prototype, 'insert', function (data, idx) { return this.slice(0, idx) + data + this.slice(idx, this.length) }) Object.defineProperty(Array.prototype, 'has', function (v) { return this.indexOf(v) !== -1 }) Object.defineProperty(Array.prototype, 'forEach', function (array, fn) { for (let i = 0; i < array.length; i++) { fn(array[i]) } }) Object.defineProperty(Array.prototype, 'map', function (fn, array) { let result = [] Array.forEach(array, function (element) { result.push(fn(element)) }) return result }) /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ utils.common = { objToString: function (obj) { try { return '' + obj } catch (e) { return '[' + e + ']' } }, getAllProperties: function (obj) { let list = [] while (obj) { list = list.concat(Object.getOwnPropertyNames(obj)) obj = Object.getPrototypeOf(obj) } return list }, getKeysFromHash: function (obj) { let list = [] for (let p in obj) { list.push(p) } return list }, quote: function (obj) { return JSON.stringify(obj) }, shuffle: function (list) { let newArray = list.slice() let len = newArray.length let i = len while (i--) { let p = parseInt(Math.random() * len) let t = newArray[i] newArray[i] = newArray[p] newArray[p] = t } return newArray }, uniqueList: function (list) { let tmp = {} let r = [] for (let i = 0; i < list.length; i++) { tmp[list[i]] = list[i] } for (let i in tmp) { r.push(tmp[i]) } return r }, mergeHash: function (obj1, obj2) { for (let p in obj2) { try { if (obj2[p].constructor === Object) { obj1[p] = utils.common.mergeHash(obj1[p], obj2[p]) } else { obj1[p] = obj2[p] } } catch (e) { obj1[p] = obj2[p] } } return obj1 } } /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ utils.block = { block: function (list, optional) { if (optional === true) { if (random.chance(6)) { return '' } } function deeper (item) { if (item === null || item === undefined) { return '' } if (typeof (item) === 'function') { return item() } if (typeof (item) === 'string') { return item } if (item instanceof (Array)) { let s = '' for (let i = 0; i < item.length; i++) { s += deeper(item[i]) } return s } return item } let asString = '' for (let i = 0; i < list.length; i++) { asString += deeper(list[i]) } return asString } } /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ var o = null // eslint-disable-line no-unused-vars function Objects () { this.counter = 0 this.container = {} } Objects.prototype.add = function (category, member) { if (!member) { member = 'o' + this.counter } if (!this.has(category)) { this.container[category] = [] } this.container[category].push({type: category, name: member}) ++this.counter return this.container[category].slice(-1)[0].name } Objects.prototype.get = function (category, last) { if (!(category in this.container)) { // return {type:null, name:null}; logger.traceback() throw new Error(category + ' is not available.') } if (last) { return this.container[category].slice(-1)[0] } return random.index(this.container[category]) } Objects.prototype.pick = function (category, last) { try { return this.get(category, last).name } catch (e) { logger.traceback() throw logger.JSError('Error: pick(' + category + ') ' + category + ' is undefined.') } } Objects.prototype.pop = function (objectName) { let self = this utils.getKeysFromHash(this.container).forEach(function (category) { self.container[category].forEach(function (obj) { if (obj.name === objectName) { self.container[category].splice(self.container[category].indexOf(obj), 1) } }) }) } Objects.prototype.contains = function (categoryNames) { let categories = [] let self = this categoryNames.forEach(function (name) { if (self.has(name)) { categories.push(name) } }) return (categories.length === 0) ? null : categories } Objects.prototype.show = function (category) { return (category in this.container) ? this.container[category] : this.container } Objects.prototype.count = function (category) { return (category in this.container) ? this.container[category].length : 0 } Objects.prototype.has = function (category) { if (category in this.container) { this.check(category) return this.container[category].length > 0 } return false } Objects.prototype.valid = function () { let items = [] let self = this utils.common.getKeysFromHash(self.container).forEach(function (category) { self.check(category) }) utils.common.getKeysFromHash(self.container).forEach(function (category) { for (let i = 0; i < self.container[category].length; i++) { items.push(self.container[category][i].name) } }) return items } Objects.prototype.check = function (category) { let self = this self.container[category].forEach(function (object) { try { let x = /* frame.contentWindow. */ eval(object.name) // eslint-disable-line no-eval if (x === undefined || x === null) { self.pop(object.name) } } catch (e) { self.pop(object.name) } }) }