octo-deno/deploy/octo.js

1704 lines
51 KiB
JavaScript

/* 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([
/* <absolute-size> values */
['xx-small', 'x-small', 'small', 'medium', 'large', 'x-large', 'xx-large'],
/* <relative-size> values */
['larger', 'smaller'],
/* <length> values */
make.number.unsignedNumber() + make.unit.unit(),
/* <percentage> 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)
}
})
}