From 748562255236360cc1b33eee6b620e967f5e5342 Mon Sep 17 00:00:00 2001 From: xpl Date: Tue, 16 Jan 2018 23:06:01 +0300 Subject: [PATCH 01/38] added functions/platform.js --- js/base/functions/platform.js | 11 +++++++++++ 1 file changed, 11 insertions(+) create mode 100644 js/base/functions/platform.js diff --git a/js/base/functions/platform.js b/js/base/functions/platform.js new file mode 100644 index 0000000000000..177c8d06b2578 --- /dev/null +++ b/js/base/functions/platform.js @@ -0,0 +1,11 @@ +"use strict"; + +/* ------------------------------------------------------------------------ */ + +module.exports = { + + isNode: (typeof window === 'undefined') && + !((typeof WorkerGlobalScope !== 'undefined') && (self instanceof WorkerGlobalScope)) +} + +/* ------------------------------------------------------------------------ */ From fee615ab8a4bf346d5c25e806296d5fe1adf98fd Mon Sep 17 00:00:00 2001 From: xpl Date: Tue, 16 Jan 2018 23:06:23 +0300 Subject: [PATCH 02/38] added functions/generic.js --- js/base/functions/generic.js | 161 +++++++++++++++++++++++++++++++++++ 1 file changed, 161 insertions(+) create mode 100644 js/base/functions/generic.js diff --git a/js/base/functions/generic.js b/js/base/functions/generic.js new file mode 100644 index 0000000000000..24cfd3e9816e5 --- /dev/null +++ b/js/base/functions/generic.js @@ -0,0 +1,161 @@ +"use strict"; + +/* ------------------------------------------------------------------------ */ + +const empty = () => Object.create (null) // empty obj without even a prototype + + , keys = Object.keys + + , values = x => !Array.isArray (x) // don't copy arrays if they're already arrays! + ? Object.values (x) + : x + + , index = x => new Set (values (x)) + + , extend = (...args) => Object.assign (empty (), ...args) // NB: side-effect free + + , clone = x => Array.isArray (x) + ? Array.from (x) // clones arrays + : extend (x) // clones objects + +/* ------------------------------------------------------------------------ */ + +module.exports = + + { empty + , keys + , values + , extend + , clone + , index + , ordered: x => x // a stub to keep assoc keys in order (in JS it does nothing, it's mostly for Python) + , unique: x => Array.from (index (x)) + +/* ------------------------------------------------------------------------ */ + + , keysort (x, out = empty ()) { + + for (const k of keys (x).sort ()) + out[k] = x[k] + + return out + } + +/* ------------------------------------------------------------------------ */ + + , indexBy (x, k, out = empty ()) { + + for (const v in values (x)) + if (key in v) + out[v[k]] = v + + return out + } + +/* ------------------------------------------------------------------------ */ + + , groupBy (x, k, out = empty ()) { + + for (const v in values (x)) { + if (k in v) { + const p = v[k] + out[p] = out[p] || [] + out[p].push (v) + } + } + return out + } + +/* ------------------------------------------------------------------------ */ + + , filterBy (x, k, v = undefined, out = []) { + + for (const v in values (x)) + if (v[k] === v) + out.push (v) + + return out + } + +/* ------------------------------------------------------------------------ */ + + , sortBy: (array, // NB: MUTATES ARRAY! + key, + descending = false, + direction = descending ? -1 : 1) => array.sort ((a, b) => + ((a[key] < b[key]) ? -direction : + ((a[key] > b[key]) ? direction : 0))) + +/* ------------------------------------------------------------------------ */ + + , flatten: function flatten (x, out = []) { + + for (const v of x) { + if (Array.isArray (v)) flatten (v, out) + else out.push (v) + } + + return out + } + +/* ------------------------------------------------------------------------ */ + + , pluck: (x, k) => values (x) + .filter (v => k in v) + .map (v => v[k]) + +/* ------------------------------------------------------------------------ */ + + , omit (x, ...args) { + + const out = clone (x) + + for (const k of args) { + + if (typeof k === 'string') // omit (x, 'a', 'b') + delete out[k] + + else if (Array.isArray (k)) // omit (x, ['a', 'b']) + for (const kk of k) + delete out[kk] + } + + return out + } + +/* ------------------------------------------------------------------------ */ + + , sum (...xs) { + + const ns = xs.filter (Number.isFinite) // leave only numbers + + return (ns.length > 0) + ? ns.reduce ((a, b) => a + b, 0) + : undefined + } + +/* ------------------------------------------------------------------------ */ + + , deepExtend: function deepExtend (...xs) { + + let out = undefined + + for (const x of xs) { + + if (x && (typeof x === 'object') && !Array.isArray (x)) { + + if (typeof out !== 'object') + out = Object.create (null) + + for (const k in x) + out[k] = deepExtend (out[k], x[k]) + + } else out = x + } + + return out + } + +/* ------------------------------------------------------------------------ */ + +} From d0346ebc43c6dd05f1263401d57ebcb5cec386ab Mon Sep 17 00:00:00 2001 From: xpl Date: Tue, 16 Jan 2018 23:06:42 +0300 Subject: [PATCH 03/38] added functions/string.js --- js/base/functions/string.js | 17 +++++++++++++++++ 1 file changed, 17 insertions(+) create mode 100644 js/base/functions/string.js diff --git a/js/base/functions/string.js b/js/base/functions/string.js new file mode 100644 index 0000000000000..d29acfb885329 --- /dev/null +++ b/js/base/functions/string.js @@ -0,0 +1,17 @@ +"use strict"; + +/* ------------------------------------------------------------------------ */ + +module.exports = + + { uuid: a => a ? (a ^ Math.random () * 16 >> a / 4).toString (16) + : ([1e7]+-1e3+-4e3+-8e3+-1e11).replace (/[018]/g, uuid) + + , unCamelCase: s => s.replace (/[a-z][A-Z]/g, x => x[0] + '_' + x[1].toLowerCase ()) // fromCamelCase → from_camel_case + + , capitalize: s => s.length + ? (s.charAt (0).toUpperCase () + s.slice (1)) + : s + } + +/* ------------------------------------------------------------------------ */ From b2bd20de100f93610121f49e28406eb7204acb0f Mon Sep 17 00:00:00 2001 From: xpl Date: Tue, 16 Jan 2018 23:06:51 +0300 Subject: [PATCH 04/38] added functions/safe.js --- js/base/functions/safe.js | 24 ++++++++++++++++++++++++ 1 file changed, 24 insertions(+) create mode 100644 js/base/functions/safe.js diff --git a/js/base/functions/safe.js b/js/base/functions/safe.js new file mode 100644 index 0000000000000..c842a862ab52f --- /dev/null +++ b/js/base/functions/safe.js @@ -0,0 +1,24 @@ +"use strict"; + +/* ------------------------------------------------------------------------ */ + +module.exports = { + + safeFloat: (obj, key, default_ = undefined, n = parseFloat (obj && obj[key])) => Number.isFinite (n) ? n : default_, + safeInteger: (obj, key, default_ = undefined, n = parseInt (obj && obj[key], 10)) => Number.isFinite (n) ? n : default_, + safeValue: (obj, key, default_ = undefined, x = obj && obj[key]) => ((x !== undefined) && (x !== null)) ? x : default_, + safeString: (obj, key, default_ = undefined) => { + + if (!obj || !(key in obj)) + return default_ + + const x = obj[key] + + if (!x && (typeof x !== 'string') && !Number.isFinite (x)) + return default_ + + return x.toString () + } +} + +/* ------------------------------------------------------------------------ */ From b89c252b0ce963a5015d83ffde1b04db6067fcd4 Mon Sep 17 00:00:00 2001 From: xpl Date: Tue, 16 Jan 2018 23:07:02 +0300 Subject: [PATCH 05/38] added functions/number.js --- js/base/functions/number.js | 66 +++++++++++++++++++++++++++++++++++++ 1 file changed, 66 insertions(+) create mode 100644 js/base/functions/number.js diff --git a/js/base/functions/number.js b/js/base/functions/number.js new file mode 100644 index 0000000000000..0de1cae00f3e0 --- /dev/null +++ b/js/base/functions/number.js @@ -0,0 +1,66 @@ +"use strict"; + +/* ------------------------------------------------------------------------ */ + +const decimal = float => parseFloat (float).toString () + +/* ------------------------------------------------------------------------ */ + +//toPrecision (x, { round: true, digits: 6, fixed: true, pad: true, output: 'string' }) + +// See https://stackoverflow.com/questions/1685680/how-to-avoid-scientific-notation-for-large-numbers-in-javascript for discussion + +function toFixed (x) { // avoid scientific notation for too large and too small numbers + + if (Math.abs (x) < 1.0) { + const e = parseInt (x.toString ().split ('e-')[1]) + if (e) { + x *= Math.pow (10, e-1) + x = '0.' + (new Array (e)).join ('0') + x.toString ().substring (2) + } + } else { + let e = parseInt (x.toString ().split ('+')[1]) + if (e > 20) { + e -= 20 + x /= Math.pow (10, e) + x += (new Array (e+1)).join ('0') + } + } + return x +} + +const numberToString = toFixed + +// See https://stackoverflow.com/questions/4912788/truncate-not-round-off-decimal-numbers-in-javascript for discussion + +// > So, after all it turned out, rounding bugs will always haunt you, no matter how hard you try to compensate them. +// > Hence the problem should be attacked by representing numbers exactly in decimal notation. + +const truncate_regExpCache = [] + , truncate_to_string = (num, precision = 0) => { + num = toFixed (num) + if (precision > 0) { + const re = truncate_regExpCache[precision] || (truncate_regExpCache[precision] = new RegExp("([-]*\\d+\\.\\d{" + precision + "})(\\d)")) + const [,result] = num.toString ().match (re) || [null, num] + return result.toString () + } + return parseInt (num).toString () + } + , truncate = (num, precision = 0) => parseFloat (truncate_to_string (num, precision)) + +const precisionFromString = (string) => { + const split = string.replace (/0+$/g, '').split ('.') + return (split.length > 1) ? (split[1].length) : 0 +} + +/* ------------------------------------------------------------------------ */ + +module.exports = { + + decimal, + toFixed, + truncate_to_string, + precisionFromString +} + +/* ------------------------------------------------------------------------ */ From 270e180baf863439cafae7664fa16be67a3b0cb4 Mon Sep 17 00:00:00 2001 From: xpl Date: Tue, 16 Jan 2018 23:07:12 +0300 Subject: [PATCH 06/38] added functions/encode.js --- js/base/functions/encode.js | 35 +++++++++++++++++++++++++++++++++++ 1 file changed, 35 insertions(+) create mode 100644 js/base/functions/encode.js diff --git a/js/base/functions/encode.js b/js/base/functions/encode.js new file mode 100644 index 0000000000000..842ec1d480276 --- /dev/null +++ b/js/base/functions/encode.js @@ -0,0 +1,35 @@ +"use strict"; + +/* ------------------------------------------------------------------------ */ + +const qs = require ('qs') // querystring (TODO: get rid of that dependency) + +/* ------------------------------------------------------------------------ */ + +module.exports = { + + stringToBinary (str) { + const arr = new Uint8Array (str.length) + for (let i = 0; i < str.length; i++) { arr[i] = str.charCodeAt(i); } + return CryptoJS.lib.WordArray.create (arr) + } + + , stringToBase64: string => CryptoJS.enc.Latin1.parse (string).toString (CryptoJS.enc.Base64) + , utf16ToBase64: string => CryptoJS.enc.Utf16 .parse (string).toString (CryptoJS.enc.Base64) + , base64ToBinary: string => CryptoJS.enc.Base64.parse (string) + , base64ToString: string => CryptoJS.enc.Base64.parse (string).toString (CryptoJS.enc.Utf8) + , binaryToString: string => string + + , binaryConcat: (...args) => args.reduce ((a, b) => a.concat (b)) + + , urlencode = object => qs.stringify (object) + , rawencode = object => qs.stringify (object, { encode: false }) + + // Url-safe-base64 without equals signs, with + replaced by - and slashes replaced by underscores + + , urlencodeBase64: base64string => base64string.replace (/[=]+$/, '') + .replace (/\+/g, '-') + .replace (/\//g, '_') +} + +/* ------------------------------------------------------------------------ */ From 8bcc5fb39c29d1dc9b8c7dd2fbcf2f5cab469fcf Mon Sep 17 00:00:00 2001 From: xpl Date: Tue, 16 Jan 2018 23:07:21 +0300 Subject: [PATCH 07/38] added functions/crypto.js --- js/base/functions/crypto.js | 38 +++++++++++++++++++++++++++++++++++++ 1 file changed, 38 insertions(+) create mode 100644 js/base/functions/crypto.js diff --git a/js/base/functions/crypto.js b/js/base/functions/crypto.js new file mode 100644 index 0000000000000..5c8f2596dac1d --- /dev/null +++ b/js/base/functions/crypto.js @@ -0,0 +1,38 @@ +"use strict"; + +/* ------------------------------------------------------------------------ */ + +const CryptoJS = require ('crypto-js') +const { capitalize } = require ('./generic') +const { stringToBase64, utf16ToBase64, urlencodeBase64 } = require ('./encode') + +/* ------------------------------------------------------------------------ */ + +const hash = (request, hash = 'md5', digest = 'hex') => { + const result = CryptoJS[hash.toUpperCase ()] (request) + return (digest == 'binary') ? result : result.toString (CryptoJS.enc[capitalize (digest)]) +} + +const hmac = (request, secret, hash = 'sha256', digest = 'hex') => { + const encoding = (digest == 'binary') ? 'Latin1' : capitalize (digest) + return CryptoJS['Hmac' + hash.toUpperCase ()] (request, secret).toString (CryptoJS.enc[capitalize (encoding)]) +} + +const jwt = function JSON_web_token (request, secret, alg = 'HS256', hash = 'sha256') { + const encodedHeader = urlencodeBase64 (stringToBase64 (JSON.stringify ({ 'alg': alg, 'typ': 'JWT' }))) + , encodedData = urlencodeBase64 (stringToBase64 (JSON.stringify (request))) + , token = [ encodedHeader, encodedData ].join ('.') + , signature = urlencodeBase64 (utf16ToBase64 (hmac (token, secret, hash, 'utf16'))) + return [ token, signature ].join ('.') +} + +/* ------------------------------------------------------------------------ */ + +module.exports = { + + hash, + hmac, + jwt +} + +/* ------------------------------------------------------------------------ */ From 1418c8d39d9c1b777fa7eda2ba0663a216353dda Mon Sep 17 00:00:00 2001 From: xpl Date: Tue, 16 Jan 2018 23:07:29 +0300 Subject: [PATCH 08/38] added functions/time.js --- js/base/functions/time.js | 78 +++++++++++++++++++++++++++++++++++++++ 1 file changed, 78 insertions(+) create mode 100644 js/base/functions/time.js diff --git a/js/base/functions/time.js b/js/base/functions/time.js new file mode 100644 index 0000000000000..9c82558d19ba5 --- /dev/null +++ b/js/base/functions/time.js @@ -0,0 +1,78 @@ +"use strict"; + +/* ------------------------------------------------------------------------ */ + +const { isNode } = require ('./platform') + +/* ------------------------------------------------------------------------ */ + +const time = isNode ? (require ('perf_hooks').performance) : // a built-in high-resolution timer available in Node + (typeof performance !== 'undefined') ? performance // ...it is also should be available in modern browsers + : Date // (fall back to the default standard resolution timer) + +/* ------------------------------------------------------------------------ */ + +const setTimeout_safe = (done, ms, setTimeout = setTimeout_original /* overrideable for mocking purposes */, targetTime = time.now () + ms) => { + +/* The built-in setTimeout function can fire its callback earlier than specified, so we + need to ensure that it does not happen: sleep recursively until `targetTime` is reached... */ + + let clearInnerTimeout = () => {} + let active = true + + let id = setTimeout (() => { + active = true + const rest = targetTime - time.now () + if (rest > 0) { + clearInnerTimeout = setTimeout_safe (done, rest, setTimeout, targetTime) // try sleep more + } else { + done () + } + }, ms) + + return function clear () { + if (active) { + active = false // dunno if IDs are unique on various platforms, so it's better to rely on this flag to exclude the possible cancellation of the wrong timer (if called after completion) + clearTimeout (id) + } + clearInnerTimeout () + } +} + +/* ------------------------------------------------------------------------ */ + +class TimedOut extends Error { + + constructor () { + const message = 'timed out' + super (message) + this.constructor = Error + this.__proto__ = Error.prototype + this.message = message + } +} + +/* ------------------------------------------------------------------------ */ + +module.exports = { + + time, + + sleep: ms => new Promise (resolve => setTimeout_safe (resolve, ms)), + + TimedOut, + + async timeout (ms, promise) { + + let clear = () => {} + const expires = new Promise (resolve => (clear = setTimeout_safe (resolve, ms))) + + try { + return await Promise.race ([promise, expires.then (() => { throw new TimedOut () })]) + } finally { + clear () // fixes https://github.com/ccxt/ccxt/issues/749 + } + } +} + +/* ------------------------------------------------------------------------ */ From ea24ea2cf909592f3e3b24b04804bafcb4c78980 Mon Sep 17 00:00:00 2001 From: xpl Date: Tue, 16 Jan 2018 23:08:31 +0300 Subject: [PATCH 09/38] linked functions.js to its sub-modules, added automatic camelCase to camel_case conversion for its methods --- js/base/functions.js | 401 ++++--------------------------------------- 1 file changed, 33 insertions(+), 368 deletions(-) diff --git a/js/base/functions.js b/js/base/functions.js index 3b90fa78cf410..082825fcfd728 100644 --- a/js/base/functions.js +++ b/js/base/functions.js @@ -1,383 +1,48 @@ "use strict"; -//----------------------------------------------------------------------------- +/* ------------------------------------------------------------------------ */ -const CryptoJS = require ('crypto-js') - , qs = require ('qs') // querystring +const { unCamelCase } = require ('./function/string') -//----------------------------------------------------------------------------- +const unCamelCasePropertyNames = x => { + for (const k in x) + x[unCamelCase (k)] = x[k] // camel_case_method = camelCaseMethod + } -const { RequestTimeout } = require ('./errors') +/* ------------------------------------------------------------------------ */ -//----------------------------------------------------------------------------- -// utility helpers +module.exports = unCamelCasePropertyNames ({ -const setTimeout_original = setTimeout + ...require ('./functions/platform'), + ...require ('./functions/generic'), + ...require ('./functions/string'), + ...require ('./functions/safe'), + ...require ('./functions/number'), + ...require ('./functions/encode'), + ...require ('./functions/crypto'), + ...require ('./functions/time'), -// setTimeout can fire earlier than specified, so we need to ensure it does not happen... +/* ------------------------------------------------------------------------ */ -const setTimeout_safe = (done, ms, setTimeout = setTimeout_original /* overrideable for mocking purposes */, targetTime = Date.now () + ms) => { - - let clearInnerTimeout = () => {} - let active = true - - let id = setTimeout (() => { - active = true - const rest = targetTime - Date.now () - if (rest > 0) { - clearInnerTimeout = setTimeout_safe (done, rest, setTimeout, targetTime) // try sleep more - } else { - done () - } - }, ms) - - return function clear () { - if (active) { - active = false // dunno if IDs are unique on various platforms, so it's better to rely on this flag to exclude the possible cancellation of the wrong timer (if called after completion) - clearTimeout (id) - } - clearInnerTimeout () - } -} - -const sleep = ms => new Promise (resolve => setTimeout_safe (resolve, ms)) - -const decimal = float => parseFloat (float).toString () - -const timeout = async (ms, promise) => { - - let clear = () => {} - const timeout = new Promise (resolve => (clear = setTimeout_safe (resolve, ms))) - - try { - return await Promise.race ([promise, timeout.then (() => { throw new RequestTimeout ('request timed out') })]) - } finally { - clear () // fixes https://github.com/ccxt/ccxt/issues/749 - } -} - -const capitalize = string => string.length ? (string.charAt (0).toUpperCase () + string.slice (1)) : string - -const keysort = object => { - const result = {} - Object.keys (object).sort ().forEach (key => result[key] = object[key]) - return result -} - -const extend = (...args) => Object.assign ({}, ...args) - -const deepExtend = function (...args) { - - // if (args.length < 1) - // return args - // else if (args.length < 2) - // return args[0] - - let result = undefined - - for (const arg of args) { - - if (arg && (typeof arg === 'object') && (arg.constructor === Object || !('constructor' in arg))) { - - if (typeof result !== 'object') { - result = {} - } - - for (const key in arg) { - result[key] = deepExtend (result[key], arg[key]) - } - - } else { - - result = arg - } - } + json: JSON.stringify, + unjson: JSON.parse, - return result -} +/* ------------------------------------------------------------------------ */ -const omit = (object, ...args) => { - const result = extend (object) - for (const x of args) { - if (typeof x === 'string') { - delete result[x] - } else if (Array.isArray (x)) { - for (const k of x) - delete result[k] - } - } - return result -} + aggregate (bidasks) { // orderbook aggregation helper -const groupBy = (array, key) => { - const result = {} - Object - .values (array) - .filter (entry => typeof entry[key] != 'undefined') - .forEach (entry => { - if (typeof result[entry[key]] == 'undefined') - result[entry[key]] = [] - result[entry[key]].push (entry) + let result = {} + + bidasks.forEach (([ price, volume ]) => { + if (volume > 0) + result[price] = (result[price] || 0) + volume }) - return result -} - -const filterBy = (array, key, value = undefined) => { - if (value) { - let grouped = groupBy (array, key) - if (value in grouped) - return grouped[value] - return [] + + return Object.keys (result).map (price => [ + parseFloat (price), + parseFloat (result[price]), + ]) } - return array -} - -const indexBy = (array, key) => { - const result = {} - Object - .values (array) - .filter (entry => typeof entry[key] != 'undefined') - .forEach (entry => { - result[entry[key]] = entry - }) - return result -} - -const sortBy = (array, key, descending = false) => { - descending = descending ? -1 : 1 - return array.sort ((a, b) => ((a[key] < b[key]) ? -descending : ((a[key] > b[key]) ? descending : 0))) -} - -const flatten = (array, result = []) => { - for (let i = 0, length = array.length; i < length; i++) { - const value = array[i] - if (Array.isArray (value)) { - flatten (value, result) - } else { - result.push (value) - } - } - return result -} - -const unique = array => array.filter ((value, index, self) => (self.indexOf (value) == index)) - -const pluck = (array, key) => array - .filter (element => (typeof element[key] != 'undefined')) - .map (element => element[key]) - -const urlencode = object => qs.stringify (object) -const rawencode = object => qs.stringify (object, { encode: false }) - -const sum = (...args) => { - const result = args.filter (arg => typeof arg !== 'undefined') - return (result.length > 0) ? - result.reduce ((sum, value) => sum + value, 0) : undefined -} - -const safeFloat = (object, key, defaultValue = undefined) => { - if (key in object) { - if (typeof object[key] == 'number') - return object[key] - else if ((typeof object[key] == 'string') && object[key]) - return parseFloat (object[key]) - } - return defaultValue -} - -const safeString = (object, key, defaultValue = undefined) => { - if (!object || !(key in object)) - return defaultValue; - let stringVal = object[key]; - if (!stringVal && typeof stringVal != 'string' && typeof stringVal != 'number') - return defaultValue; - return stringVal.toString (); -} +}) -const safeInteger = (object, key, defaultValue = undefined) => { - if (!object || !(key in object)) - return defaultValue; - let intVal = parseInt (object[key], 10); - return isNaN (intVal) ? defaultValue : intVal; -} - -const safeValue = (object, key, defaultValue = undefined) => { - return (object && (key in object) && object[key]) ? object[key] : defaultValue -} - -const uuid = a => a ? - (a ^ Math.random () * 16 >> a / 4).toString (16) : - ([1e7]+-1e3+-4e3+-8e3+-1e11).replace (/[018]/g, uuid) - -// See https://stackoverflow.com/questions/1685680/how-to-avoid-scientific-notation-for-large-numbers-in-javascript for discussion - -function toFixed (x) { // avoid scientific notation for too large and too small numbers - - if (Math.abs (x) < 1.0) { - const e = parseInt (x.toString ().split ('e-')[1]) - if (e) { - x *= Math.pow (10, e-1) - x = '0.' + (new Array (e)).join ('0') + x.toString ().substring (2) - } - } else { - let e = parseInt (x.toString ().split ('+')[1]) - if (e > 20) { - e -= 20 - x /= Math.pow (10, e) - x += (new Array (e+1)).join ('0') - } - } - return x -} - -// See https://stackoverflow.com/questions/4912788/truncate-not-round-off-decimal-numbers-in-javascript for discussion - -// > So, after all it turned out, rounding bugs will always haunt you, no matter how hard you try to compensate them. -// > Hence the problem should be attacked by representing numbers exactly in decimal notation. - -const truncate_regExpCache = [] - , truncate_to_string = (num, precision = 0) => { - num = toFixed (num) - if (precision > 0) { - const re = truncate_regExpCache[precision] || (truncate_regExpCache[precision] = new RegExp("([-]*\\d+\\.\\d{" + precision + "})(\\d)")) - const [,result] = num.toString ().match (re) || [null, num] - return result.toString () - } - return parseInt (num).toString () - } - , truncate = (num, precision = 0) => parseFloat (truncate_to_string (num, precision)) - -const precisionFromString = (string) => { - const split = string.replace (/0+$/g, '').split ('.') - return (split.length > 1) ? (split[1].length) : 0 -} - -const ordered = x => x // a stub to keep assoc keys in order, in JS it does nothing, it's mostly for Python - -const aggregate = function (bidasks) { - - let result = {} - - bidasks.forEach (([ price, volume ]) => { - if (volume > 0) - result[price] = (result[price] || 0) + volume - }) - - return Object.keys (result).map (price => [ - parseFloat (price), - parseFloat (result[price]), - ]) -} - -//----------------------------------------------------------------------------- -// string ←→ binary ←→ base64 conversion routines - -const stringToBinary = str => { - const arr = new Uint8Array (str.length) - for (let i = 0; i < str.length; i++) { arr[i] = str.charCodeAt(i); } - return CryptoJS.lib.WordArray.create (arr) -} - -const stringToBase64 = string => CryptoJS.enc.Latin1.parse (string).toString (CryptoJS.enc.Base64) - , utf16ToBase64 = string => CryptoJS.enc.Utf16 .parse (string).toString (CryptoJS.enc.Base64) - , base64ToBinary = string => CryptoJS.enc.Base64.parse (string) - , base64ToString = string => CryptoJS.enc.Base64.parse (string).toString (CryptoJS.enc.Utf8) - , binaryToString = string => string - -const binaryConcat = (...args) => args.reduce ((a, b) => a.concat (b)) - -// url-safe-base64 without equals signs, with + replaced by - and slashes replaced by underscores -const urlencodeBase64 = base64string => base64string.replace (/[=]+$/, '') - .replace (/\+/g, '-') - .replace (/\//g, '_') - -//----------------------------------------------------------------------------- -// cryptography - -const hash = (request, hash = 'md5', digest = 'hex') => { - const result = CryptoJS[hash.toUpperCase ()] (request) - return (digest == 'binary') ? result : result.toString (CryptoJS.enc[capitalize (digest)]) -} - -const hmac = (request, secret, hash = 'sha256', digest = 'hex') => { - const encoding = (digest == 'binary') ? 'Latin1' : capitalize (digest) - return CryptoJS['Hmac' + hash.toUpperCase ()] (request, secret).toString (CryptoJS.enc[capitalize (encoding)]) -} - -//----------------------------------------------------------------------------- -// a JSON Web Token authentication method - -const jwt = (request, secret, alg = 'HS256', hash = 'sha256') => { - const encodedHeader = urlencodeBase64 (stringToBase64 (JSON.stringify ({ 'alg': alg, 'typ': 'JWT' }))) - , encodedData = urlencodeBase64 (stringToBase64 (JSON.stringify (request))) - , token = [ encodedHeader, encodedData ].join ('.') - , signature = urlencodeBase64 (utf16ToBase64 (hmac (token, secret, hash, 'utf16'))) - return [ token, signature ].join ('.') -} - -//----------------------------------------------------------------------------- - -module.exports = { - - setTimeout_safe, - - // common utility functions - - sleep, - timeout, - capitalize, - keysort, - extend, - deepExtend, - omit, - groupBy, - indexBy, - sortBy, - filterBy, - flatten, - unique, - pluck, - urlencode, - rawencode, - sum, - decimal, - safeFloat, - safeString, - safeInteger, - safeValue, - ordered, - aggregate, - truncate, - truncate_to_string, - uuid, - precisionFromString, - - // underscore aliases - - index_by: indexBy, - sort_by: sortBy, - group_by: groupBy, - filter_by: filterBy, - safe_float: safeFloat, - safe_string: safeString, - safe_integer: safeInteger, - safe_value: safeValue, - - // crypto functions - - binaryConcat, - stringToBinary, - binaryToString, - stringToBase64, - utf16ToBase64, - base64ToBinary, - base64ToString, - urlencodeBase64, - hash, - hmac, - jwt, - - // json - json: JSON.stringify, - unjson: JSON.parse -} +/* ------------------------------------------------------------------------ */ From 5b784ed5341b6ccf53850ad3eb6ae19f4d38b298 Mon Sep 17 00:00:00 2001 From: xpl Date: Tue, 16 Jan 2018 23:09:23 +0300 Subject: [PATCH 10/38] Exchange.js integrating refactored stuff WIP --- js/base/Exchange.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/js/base/Exchange.js b/js/base/Exchange.js index 6bf2913fe7ea7..317723532d572 100644 --- a/js/base/Exchange.js +++ b/js/base/Exchange.js @@ -291,8 +291,8 @@ module.exports = class Exchange { .then (response => this.handleRestResponse (response, url, method, headers, body)) return timeout (this.timeout, promise).catch (e => { - if (e instanceof RequestTimeout) - throw new RequestTimeout (this.id + ' ' + method + ' ' + url + ' ' + e.message + ' (' + this.timeout + ' ms)') + if (e instanceof TimedOut) + throw new RequestTimeout (this.id + ' ' + method + ' ' + url + ' request timed out (' + this.timeout + ' ms)') throw e }) } From 507d1fc10b263a3bfa5b580f320890e1ee059ca7 Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 00:27:20 +0300 Subject: [PATCH 11/38] moved throttle to functions/throttle.js --- js/base/{ => functions}/throttle.js | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename js/base/{ => functions}/throttle.js (100%) diff --git a/js/base/throttle.js b/js/base/functions/throttle.js similarity index 100% rename from js/base/throttle.js rename to js/base/functions/throttle.js From dec22a17693d8429bfc2872e99fecc3872f3475b Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 00:31:19 +0300 Subject: [PATCH 12/38] error class hierarchy now fully generated in run-time in JS from a succint tree schema... DRY --- js/base/errors.js | 180 ++++++++++++++-------------------------------- 1 file changed, 53 insertions(+), 127 deletions(-) diff --git a/js/base/errors.js b/js/base/errors.js index ebb5920204c10..6d792420fa94e 100644 --- a/js/base/errors.js +++ b/js/base/errors.js @@ -1,144 +1,70 @@ -class BaseError extends Error { - constructor (message) { - super (message) - // a workaround to make `instanceof BaseError` work in ES5 - this.constructor = BaseError - this.__proto__ = BaseError.prototype - this.message = message - } -} +/* ------------------------------------------------------------------------ */ -class ExchangeError extends BaseError { - constructor (message) { - super (message) - this.constructor = ExchangeError - this.__proto__ = ExchangeError.prototype - this.message = message - } -} +module.exports = subclass ( -class NotSupported extends ExchangeError { - constructor (message) { - super (message) - this.constructor = NotSupported - this.__proto__ = NotSupported.prototype - this.message = message - } -} +/* Root class */ -class AuthenticationError extends ExchangeError { - constructor (message) { - super (message) - this.constructor = AuthenticationError - this.__proto__ = AuthenticationError.prototype - this.message = message - } -} + Error, -class InvalidNonce extends ExchangeError { - constructor (message) { - super (message) - this.constructor = InvalidNonce - this.__proto__ = InvalidNonce.prototype - this.message = message - } -} +/* Derived class hierarchy */ -class InsufficientFunds extends ExchangeError { - constructor (message) { - super (message) - this.constructor = InsufficientFunds - this.__proto__ = InsufficientFunds.prototype - this.message = message + { 'BaseError': + { 'ExchangeError': + { 'NotSupported': {} + , 'AuthenticationError': {} + , 'InvalidNonce': {} + , 'InsufficientFunds': {} + , 'InvalidOrder': + { 'OrderNotFound': {} + , 'OrderNotCached': {} + , 'CancelPending': {} + } + , 'NetworkError': + { 'DDoSProtection': {} + , 'RequestTimeout': {} + , 'ExchangeNotAvailable': {} + } + } + } } -} +) -class InvalidOrder extends ExchangeError { - constructor (message) { - super (message) - this.constructor = InvalidOrder - this.__proto__ = InvalidOrder.prototype - this.message = message - } -} +/* ------------------------------------------------------------------------ */ -class OrderNotFound extends InvalidOrder { - constructor (message) { - super (message) - this.constructor = OrderNotFound - this.__proto__ = OrderNotFound.prototype - this.message = message - } -} +function subclass (BaseClass, classes, namespace = {}) { -class OrderNotCached extends InvalidOrder { - constructor (message) { - super (message) - this.constructor = OrderNotCached - this.__proto__ = OrderNotCached.prototype - this.message = message - } -} + for (const [$class, subclasses] of Object.entries (classes)) { -class CancelPending extends InvalidOrder { - constructor (message) { - super (message) - this.constructor = CancelPending - this.__proto__ = CancelPending.prototype - this.message = message - } -} + const Class = Object.assign (namespace, { + + /* By creating a named property, we trick compiler to assign our class constructor function a name. + Otherwise, all our error constructors would be shown as [Function: Error] in the debugger! And + the super-useful `e.constructor.name` magic wouldn't work — we then would have no chance to + obtain a error type string from an error instance programmatically! */ -class NetworkError extends BaseError { - constructor (message) { - super (message) - this.constructor = NetworkError - this.__proto__ = NetworkError.prototype - this.message = message - } -} + [$class]: class extends BaseClass { -class DDoSProtection extends NetworkError { - constructor (message) { - super (message) - this.constructor = DDoSProtection - this.__proto__ = DDoSProtection.prototype - this.message = message - } -} + constructor (message) { + + super (message) + + /* A workaround to make `instanceof` work on custom Error classes in transpiled ES5. + See my blog post for the explanation of this hack: + + https://medium.com/@xpl/javascript-deriving-from-error-properly-8d2f8f315801 */ + + this.constructor = Class + this.__proto__ = Class.prototype + this.message = message + } + } -class RequestTimeout extends NetworkError { - constructor (message) { - super (message) - this.constructor = RequestTimeout - this.__proto__ = RequestTimeout.prototype - this.message = message - } -} + })[$class] -class ExchangeNotAvailable extends NetworkError { - constructor (message) { - super (message) - this.constructor = ExchangeNotAvailable - this.__proto__ = ExchangeNotAvailable.prototype - this.message = message + subclass (Class, subclasses, namespace) } -} - -module.exports = { - BaseError, - ExchangeError, - NotSupported, - AuthenticationError, - InvalidNonce, - InsufficientFunds, - InvalidOrder, - OrderNotFound, - OrderNotCached, - CancelPending, - NetworkError, - DDoSProtection, - RequestTimeout, - ExchangeNotAvailable, + return namespace } + +/* ------------------------------------------------------------------------ */ From c2e8c93422e243c2fe7afc8eba0f65c692402d81 Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 00:31:37 +0300 Subject: [PATCH 13/38] formatting fixes --- js/base/functions.js | 18 ++++---- js/base/functions/crypto.js | 4 ++ js/base/functions/generic.js | 24 +++++----- js/base/functions/throttle.js | 87 ++++++++++++++++++----------------- 4 files changed, 71 insertions(+), 62 deletions(-) diff --git a/js/base/functions.js b/js/base/functions.js index 082825fcfd728..ecc39ca5c03d1 100644 --- a/js/base/functions.js +++ b/js/base/functions.js @@ -21,27 +21,27 @@ module.exports = unCamelCasePropertyNames ({ ...require ('./functions/encode'), ...require ('./functions/crypto'), ...require ('./functions/time'), + ...require ('./functions/throttle'), -/* ------------------------------------------------------------------------ */ +/* ............................................. */ json: JSON.stringify, unjson: JSON.parse, -/* ------------------------------------------------------------------------ */ +/* ............................................. */ - aggregate (bidasks) { // orderbook aggregation helper + aggregate (bidasks) { let result = {} - bidasks.forEach (([ price, volume ]) => { + for (const [price, volume] of bidasks) { if (volume > 0) result[price] = (result[price] || 0) + volume - }) + } - return Object.keys (result).map (price => [ - parseFloat (price), - parseFloat (result[price]), - ]) + return Object.keys (result) + .map (price => [parseFloat (price), + parseFloat (result[price])]) } }) diff --git a/js/base/functions/crypto.js b/js/base/functions/crypto.js index 5c8f2596dac1d..7c93adeb9deda 100644 --- a/js/base/functions/crypto.js +++ b/js/base/functions/crypto.js @@ -13,11 +13,15 @@ const hash = (request, hash = 'md5', digest = 'hex') => { return (digest == 'binary') ? result : result.toString (CryptoJS.enc[capitalize (digest)]) } +/* ............................................. */ + const hmac = (request, secret, hash = 'sha256', digest = 'hex') => { const encoding = (digest == 'binary') ? 'Latin1' : capitalize (digest) return CryptoJS['Hmac' + hash.toUpperCase ()] (request, secret).toString (CryptoJS.enc[capitalize (encoding)]) } +/* ............................................. */ + const jwt = function JSON_web_token (request, secret, alg = 'HS256', hash = 'sha256') { const encodedHeader = urlencodeBase64 (stringToBase64 (JSON.stringify ({ 'alg': alg, 'typ': 'JWT' }))) , encodedData = urlencodeBase64 (stringToBase64 (JSON.stringify (request))) diff --git a/js/base/functions/generic.js b/js/base/functions/generic.js index 24cfd3e9816e5..2bbd0fd0e93fb 100644 --- a/js/base/functions/generic.js +++ b/js/base/functions/generic.js @@ -20,9 +20,9 @@ const empty = () => Object.create (null) // empty obj without even a prototype /* ------------------------------------------------------------------------ */ -module.exports = +module.exports = { - { empty + empty , keys , values , extend @@ -31,7 +31,7 @@ module.exports = , ordered: x => x // a stub to keep assoc keys in order (in JS it does nothing, it's mostly for Python) , unique: x => Array.from (index (x)) -/* ------------------------------------------------------------------------ */ +/* ............................................. */ , keysort (x, out = empty ()) { @@ -41,7 +41,7 @@ module.exports = return out } -/* ------------------------------------------------------------------------ */ +/* ............................................. */ , indexBy (x, k, out = empty ()) { @@ -52,7 +52,7 @@ module.exports = return out } -/* ------------------------------------------------------------------------ */ +/* ............................................. */ , groupBy (x, k, out = empty ()) { @@ -66,7 +66,7 @@ module.exports = return out } -/* ------------------------------------------------------------------------ */ +/* ............................................. */ , filterBy (x, k, v = undefined, out = []) { @@ -77,7 +77,7 @@ module.exports = return out } -/* ------------------------------------------------------------------------ */ +/* ............................................. */ , sortBy: (array, // NB: MUTATES ARRAY! key, @@ -86,7 +86,7 @@ module.exports = ((a[key] < b[key]) ? -direction : ((a[key] > b[key]) ? direction : 0))) -/* ------------------------------------------------------------------------ */ +/* ............................................. */ , flatten: function flatten (x, out = []) { @@ -98,13 +98,13 @@ module.exports = return out } -/* ------------------------------------------------------------------------ */ +/* ............................................. */ , pluck: (x, k) => values (x) .filter (v => k in v) .map (v => v[k]) -/* ------------------------------------------------------------------------ */ +/* ............................................. */ , omit (x, ...args) { @@ -123,7 +123,7 @@ module.exports = return out } -/* ------------------------------------------------------------------------ */ +/* ............................................. */ , sum (...xs) { @@ -134,7 +134,7 @@ module.exports = : undefined } -/* ------------------------------------------------------------------------ */ +/* ............................................. */ , deepExtend: function deepExtend (...xs) { diff --git a/js/base/functions/throttle.js b/js/base/functions/throttle.js index e320bd9da5473..25169a3a357c1 100644 --- a/js/base/functions/throttle.js +++ b/js/base/functions/throttle.js @@ -1,59 +1,64 @@ "use strict"; -const { sleep } = require ('./functions') +/* ------------------------------------------------------------------------ */ -const throttle = cfg => { +const { sleep + , time } = require ('./time') - let lastTimestamp = Date.now () - , numTokens = (typeof cfg.numTokens != 'undefined') ? cfg.numTokens : cfg.capacity - , queue = [] - , running = false - , counter = 0 +/* ------------------------------------------------------------------------ */ - return Object.assign (cost => { +module.exports = { + + throttle: function throttle (cfg) { - if (queue.length > cfg.maxCapacity) - throw new Error ('Backlog is over max capacity of ' + cfg.maxCapacity) + let lastTimestamp = time.now () + , numTokens = (typeof cfg.numTokens != 'undefined') ? cfg.numTokens : cfg.capacity + , running = false + , counter = 0 - return new Promise (async (resolve, reject) => { + const queue = [] - try { + return Object.assign (cost => { - queue.push ({ cost, resolve, reject }) + if (queue.length > cfg.maxCapacity) + throw new Error ('Backlog is over max capacity of ' + cfg.maxCapacity) - if (!running) { - running = true - while (queue.length > 0) { - const hasEnoughTokens = cfg.capacity ? (numTokens > 0) : (numTokens >= 0) - if (hasEnoughTokens) { - if (queue.length > 0) { - let { cost, resolve, reject } = queue[0] - cost = (cost || cfg.defaultCost) - if (numTokens >= Math.min (cost, cfg.capacity)) { - numTokens -= cost - queue.shift () - resolve () + return new Promise (async (resolve, reject) => { + + try { + queue.push ({ cost, resolve, reject }) + + if (!running) { + running = true + while (queue.length > 0) { + const hasEnoughTokens = cfg.capacity ? (numTokens > 0) : (numTokens >= 0) + if (hasEnoughTokens) { + if (queue.length > 0) { + const { cost, resolve, reject } = queue[0] + cost = (cost || cfg.defaultCost) + if (numTokens >= Math.min (cost, cfg.capacity)) { + numTokens -= cost + queue.shift () + resolve () + } } } + const now = time.now () + , elapsed = now - lastTimestamp + lastTimestamp = now + numTokens = Math.min (cfg.capacity, numTokens + elapsed * cfg.refillRate) + await sleep (cfg.delay) } - let now = Date.now () - let elapsed = now - lastTimestamp - lastTimestamp = now - numTokens = Math.min (cfg.capacity, numTokens + elapsed * cfg.refillRate) - await sleep (cfg.delay) + running = false } - running = false - } - } catch (e) { - - reject (e) - } - }) + } catch (e) { + reject (e) + } + }) - }, cfg, { - configure: newCfg => throttle (Object.assign ({}, cfg, newCfg)) - }) + }, cfg, { configure: newCfg => throttle (Object.assign ({}, cfg, newCfg)) }) + } } -module.exports = throttle +/* ------------------------------------------------------------------------ */ From 6b533964fd8c0812ead1e2b780c6b7f72fcea2a4 Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 01:53:02 +0300 Subject: [PATCH 14/38] refactored safeXXX functions totally, now separate functions/type.js module --- ccxt.js | 4 +-- js/base/Exchange.js | 28 +++++++++++---------- js/base/functions.js | 6 ++--- js/base/functions/encode.js | 4 +-- js/base/functions/generic.js | 12 ++++----- js/base/functions/number.js | 13 +++++----- js/base/functions/safe.js | 24 ------------------ js/base/functions/time.js | 17 ++++++------- js/base/functions/type.js | 45 ++++++++++++++++++++++++++++++++++ js/test/test_base.js | 47 +++++++++++++++++++++++++++--------- package.json | 4 +-- 11 files changed, 125 insertions(+), 79 deletions(-) delete mode 100644 js/base/functions/safe.js create mode 100644 js/base/functions/type.js diff --git a/ccxt.js b/ccxt.js index 00890c3f29474..e8f7ae145bb53 100644 --- a/ccxt.js +++ b/ccxt.js @@ -31,8 +31,8 @@ SOFTWARE. //----------------------------------------------------------------------------- const Exchange = require ('./js/base/Exchange') -const functions = require ('./js/base/functions') -const errors = require ('./js/base/errors') + , functions = require ('./js/base/functions') + , errors = require ('./js/base/errors') //----------------------------------------------------------------------------- // this is updated by vss.js when building diff --git a/js/base/Exchange.js b/js/base/Exchange.js index 317723532d572..469bcd3156ddb 100644 --- a/js/base/Exchange.js +++ b/js/base/Exchange.js @@ -1,14 +1,12 @@ "use strict"; -//----------------------------------------------------------------------------- +/* ------------------------------------------------------------------------ */ -const isNode = (typeof window === 'undefined') && !(typeof WorkerGlobalScope !== 'undefined' && self instanceof WorkerGlobalScope) - , functions = require ('./functions') - , throttle = require ('./throttle') - , fetchImplementation = isNode ? require ('fetch-ponyfill')().fetch : fetch +const functions = require ('./functions') , Market = require ('./Market') -const { deepExtend +const { isNode + , deepExtend , extend , sleep , timeout @@ -18,7 +16,9 @@ const { deepExtend , groupBy , aggregate , uuid - , precisionFromString } = functions + , precisionFromString + , throttle + , time } = functions const { ExchangeError , NotSupported @@ -27,9 +27,11 @@ const { ExchangeError , RequestTimeout , ExchangeNotAvailable } = require ('./errors') -// stub until we get a better solution for Webpack and React -// const journal = isNode && require ('./journal') -const journal = undefined +const fetchImplementation = isNode ? require ('fetch-ponyfill')().fetch : fetch + +const journal = undefined // isNode && require ('./journal') // stub until we get a better solution for Webpack and React + +/* ------------------------------------------------------------------------ */ module.exports = class Exchange { @@ -80,9 +82,9 @@ module.exports = class Exchange { this.iso8601 = timestamp => new Date (timestamp).toISOString () this.parse8601 = x => Date.parse (((x.indexOf ('+') >= 0) || (x.slice (-1) == 'Z')) ? x : (x + 'Z')) - this.milliseconds = Date.now - this.microseconds = () => Math.floor (this.milliseconds () * 1000) - this.seconds = () => Math.floor (this.milliseconds () / 1000) + this.milliseconds = () => Math.floor (time.now ()) + this.microseconds = () => Math.floor (time.now () * 1000) + this.seconds = () => Math.floor (time.now () / 1000) this.id = undefined // rate limiter settings diff --git a/js/base/functions.js b/js/base/functions.js index ecc39ca5c03d1..9d0846128aead 100644 --- a/js/base/functions.js +++ b/js/base/functions.js @@ -2,11 +2,11 @@ /* ------------------------------------------------------------------------ */ -const { unCamelCase } = require ('./function/string') +const { unCamelCase } = require ('./functions/string') const unCamelCasePropertyNames = x => { - for (const k in x) - x[unCamelCase (k)] = x[k] // camel_case_method = camelCaseMethod + for (const k in x) x[unCamelCase (k)] = x[k] // camel_case_method = camelCaseMethod + return x } /* ------------------------------------------------------------------------ */ diff --git a/js/base/functions/encode.js b/js/base/functions/encode.js index 842ec1d480276..621d791d80df4 100644 --- a/js/base/functions/encode.js +++ b/js/base/functions/encode.js @@ -22,8 +22,8 @@ module.exports = { , binaryConcat: (...args) => args.reduce ((a, b) => a.concat (b)) - , urlencode = object => qs.stringify (object) - , rawencode = object => qs.stringify (object, { encode: false }) + , urlencode: object => qs.stringify (object) + , rawencode: object => qs.stringify (object, { encode: false }) // Url-safe-base64 without equals signs, with + replaced by - and slashes replaced by underscores diff --git a/js/base/functions/generic.js b/js/base/functions/generic.js index 2bbd0fd0e93fb..38f8a651afd4b 100644 --- a/js/base/functions/generic.js +++ b/js/base/functions/generic.js @@ -45,8 +45,8 @@ module.exports = { , indexBy (x, k, out = empty ()) { - for (const v in values (x)) - if (key in v) + for (const v of values (x)) + if (k in v) out[v[k]] = v return out @@ -56,7 +56,7 @@ module.exports = { , groupBy (x, k, out = empty ()) { - for (const v in values (x)) { + for (const v of values (x)) { if (k in v) { const p = v[k] out[p] = out[p] || [] @@ -68,10 +68,10 @@ module.exports = { /* ............................................. */ - , filterBy (x, k, v = undefined, out = []) { + , filterBy (x, k, value = undefined, out = []) { - for (const v in values (x)) - if (v[k] === v) + for (const v of values (x)) + if (v[k] === value) out.push (v) return out diff --git a/js/base/functions/number.js b/js/base/functions/number.js index 0de1cae00f3e0..0957400c9eb0a 100644 --- a/js/base/functions/number.js +++ b/js/base/functions/number.js @@ -36,17 +36,17 @@ const numberToString = toFixed // > So, after all it turned out, rounding bugs will always haunt you, no matter how hard you try to compensate them. // > Hence the problem should be attacked by representing numbers exactly in decimal notation. -const truncate_regExpCache = [] - , truncate_to_string = (num, precision = 0) => { +const regexCache = [] + , truncateToString = (num, precision = 0) => { num = toFixed (num) if (precision > 0) { - const re = truncate_regExpCache[precision] || (truncate_regExpCache[precision] = new RegExp("([-]*\\d+\\.\\d{" + precision + "})(\\d)")) + const re = regexCache[precision] || (regexCache[precision] = new RegExp("([-]*\\d+\\.\\d{" + precision + "})(\\d)")) const [,result] = num.toString ().match (re) || [null, num] return result.toString () } - return parseInt (num).toString () + return parseInt (num, 10).toString () } - , truncate = (num, precision = 0) => parseFloat (truncate_to_string (num, precision)) + , truncate = (num, precision = 0) => parseFloat (truncateToString (num, precision)) const precisionFromString = (string) => { const split = string.replace (/0+$/g, '').split ('.') @@ -59,7 +59,8 @@ module.exports = { decimal, toFixed, - truncate_to_string, + truncate, + truncateToString, precisionFromString } diff --git a/js/base/functions/safe.js b/js/base/functions/safe.js deleted file mode 100644 index c842a862ab52f..0000000000000 --- a/js/base/functions/safe.js +++ /dev/null @@ -1,24 +0,0 @@ -"use strict"; - -/* ------------------------------------------------------------------------ */ - -module.exports = { - - safeFloat: (obj, key, default_ = undefined, n = parseFloat (obj && obj[key])) => Number.isFinite (n) ? n : default_, - safeInteger: (obj, key, default_ = undefined, n = parseInt (obj && obj[key], 10)) => Number.isFinite (n) ? n : default_, - safeValue: (obj, key, default_ = undefined, x = obj && obj[key]) => ((x !== undefined) && (x !== null)) ? x : default_, - safeString: (obj, key, default_ = undefined) => { - - if (!obj || !(key in obj)) - return default_ - - const x = obj[key] - - if (!x && (typeof x !== 'string') && !Number.isFinite (x)) - return default_ - - return x.toString () - } -} - -/* ------------------------------------------------------------------------ */ diff --git a/js/base/functions/time.js b/js/base/functions/time.js index 9c82558d19ba5..333a8c67d06fa 100644 --- a/js/base/functions/time.js +++ b/js/base/functions/time.js @@ -12,6 +12,7 @@ const time = isNode ? (require ('perf_hooks').performance) : // a /* ------------------------------------------------------------------------ */ +const setTimeout_original = setTimeout const setTimeout_safe = (done, ms, setTimeout = setTimeout_original /* overrideable for mocking purposes */, targetTime = time.now () + ms) => { /* The built-in setTimeout function can fire its callback earlier than specified, so we @@ -54,19 +55,17 @@ class TimedOut extends Error { /* ------------------------------------------------------------------------ */ -module.exports = { +module.exports = - time, - - sleep: ms => new Promise (resolve => setTimeout_safe (resolve, ms)), - - TimedOut, - - async timeout (ms, promise) { + { time + , setTimeout_safe + , sleep: ms => new Promise (resolve => setTimeout_safe (resolve, ms)) + , TimedOut + , timeout: async (ms, promise) => { let clear = () => {} const expires = new Promise (resolve => (clear = setTimeout_safe (resolve, ms))) - + try { return await Promise.race ([promise, expires.then (() => { throw new TimedOut () })]) } finally { diff --git a/js/base/functions/type.js b/js/base/functions/type.js new file mode 100644 index 0000000000000..117da4cda7f3c --- /dev/null +++ b/js/base/functions/type.js @@ -0,0 +1,45 @@ +"use strict"; + +/* ------------------------------------------------------------------------ */ + +const isNumber = Number.isFinite + , isObject = o => typeof o === 'object' + , isString = s => typeof s === 'string' + , isStringCoercible = x => (hasProps (x) && x.toString) || isNumber (x) + +/* ............................................. */ + +const hasProps = o => (o !== undefined) && + (o !== null) + + , prop = (o, k) => isObject (o) ? o[k] + : undefined + +/* ............................................. */ + +const asFloat = x => (isNumber (x) || isString (x)) ? parseFloat (x) : NaN + , asInteger = x => (isNumber (x) || isString (x)) ? parseInt (x, 10) : NaN + +/* ............................................. */ + +module.exports = + + { isNumber + , isObject + , isString + , isStringCoercible + + , hasProps + , prop + + , asFloat + , asInteger + + , safeFloat: (o, k, $default, n = asFloat (prop (o, k))) => isNumber (n) ? n : $default + , safeInteger: (o, k, $default, n = asInteger (prop (o, k))) => isNumber (n) ? n : $default + , safeValue: (o, k, $default, x = prop (o, k) ) => hasProps (x) ? x : $default + , safeString: (o, k, $default, x = prop (o, k) ) => isStringCoercible (x) ? String (x) : $default + + } + +/* ------------------------------------------------------------------------ */ diff --git a/js/test/test_base.js b/js/test/test_base.js index 7954469d6765d..fa7665ee3ef11 100644 --- a/js/test/test_base.js +++ b/js/test/test_base.js @@ -1,8 +1,11 @@ /* ------------------------------------------------------------------------ */ +global.log = require ('ololog') // for easier debugging + +/* ------------------------------------------------------------------------ */ + const ccxt = require ('../../ccxt.js') , assert = require ('assert') - , log = require ('ololog') , ansi = require ('ansicolor').nice; /* ------------------------------------------------------------------------ */ @@ -32,18 +35,38 @@ describe ('ccxt base code', () => { it ('safeFloat is robust', async () => { - assert.strictEqual (ccxt.safeFloat ({'float': '1.0'}, 'float'), 1.0) - assert.strictEqual (ccxt.safeFloat ({'float': '-1.0'}, 'float'), -1.0) - assert.strictEqual (ccxt.safeFloat ({'float': 1.0}, 'float'), 1.0) - assert.strictEqual (ccxt.safeFloat ({'float': 0}, 'float'), 0) - assert.strictEqual (ccxt.safeFloat ({'float': undefined}, 'float'), undefined) - assert.strictEqual (ccxt.safeFloat ({'float': ""}, 'float'), undefined) - assert.strictEqual (ccxt.safeFloat ({'float': ""}, 'float', 0), 0) - assert.strictEqual (ccxt.safeFloat ({}, 'float'), undefined) - assert.strictEqual (ccxt.safeFloat ({}, 'float', 0), 0) + const $default = {} + + for (const fn of ['safeFloat', 'safeInteger']) { + + log (fn, ccxt.safeFloat ({ float: [0] }, 'float')) + + assert.strictEqual (ccxt[fn] ({'x': false }, 'x', $default), $default) + assert.strictEqual (ccxt[fn] ({'x': true }, 'x', $default), $default) + assert.strictEqual (ccxt[fn] ({'x': [] }, 'x', $default), $default) + assert.strictEqual (ccxt[fn] ({'x': [0] }, 'x', $default), $default) + assert.strictEqual (ccxt[fn] ({'x': [1] }, 'x', $default), $default) + assert.strictEqual (ccxt[fn] ({'x': {} }, 'x', $default), $default) + assert.strictEqual (ccxt[fn] ({'x': Number.NaN }, 'x'), undefined) + assert.strictEqual (ccxt[fn] ({'x': Number.POSITIVE_INFINITY }, 'x'), undefined) + assert.strictEqual (ccxt[fn] ({'x': null }, 'x', undefined), undefined) + assert.strictEqual (ccxt[fn] ({'x': null }, 'x', $default), $default) + assert.strictEqual (ccxt[fn] ({'x': '1.0'}, 'x'), 1.0) + assert.strictEqual (ccxt[fn] ({'x': '-1.0'}, 'x'), -1.0) + assert.strictEqual (ccxt[fn] ({'x': 1.0}, 'x'), 1.0) + assert.strictEqual (ccxt[fn] ({'x': 0}, 'x'), 0) + assert.strictEqual (ccxt[fn] ({'x': undefined}, 'x', $default), $default) + assert.strictEqual (ccxt[fn] ({'x': ""}, 'x'), undefined) + assert.strictEqual (ccxt[fn] ({'x': ""}, 'x', 0), 0) + assert.strictEqual (ccxt[fn] ({}, 'x'), undefined) + assert.strictEqual (ccxt[fn] ({}, 'x', 0), 0) + } + + assert.strictEqual (ccxt.safeFloat ({'x': 1.59999999}, 'x'), 1.59999999) + assert.strictEqual (ccxt.safeInteger ({'x': 1.59999999}, 'x'), 1) }) - it.skip ('setTimeout_safe is working', (done) => { + it ('setTimeout_safe is working', (done) => { const start = Date.now () const calls = [] @@ -79,7 +102,7 @@ describe ('ccxt base code', () => { assert ('foo', await ccxt.timeout (200, new Promise (resolve => setTimeout (() => resolve ('foo'), 100)))) await ccxt.timeout (100, Promise.reject ('foo')).should.be.rejectedWith ('foo') - await ccxt.timeout (100, new Promise ((resolve, reject) => setTimeout (() => reject ('foo'), 200))).should.be.rejectedWith ('request timed out') + await ccxt.timeout (100, new Promise ((resolve, reject) => setTimeout (() => reject ('foo'), 200))).should.be.rejectedWith ('timed out') }) it ('calculateFee() works', () => { diff --git a/package.json b/package.json index 99642d52c1061..a2962c6084958 100644 --- a/package.json +++ b/package.json @@ -15,7 +15,7 @@ "build": "npm run export-exchanges && npm run vss && npm run pandoc-python-readme && npm run pandoc-doc-readme && npm run pandoc-doc-manual && npm run pandoc-doc-install && npm run pandoc-doc-exchanges && npm run pandoc-doc-exchanges-by-country && npm run transpile && npm run qa && npm run update-badges && npm run browserify", "test": "npm run build && node run-tests", "fast-test": "node run-tests --js", - "test-base": "mocha --reporter spec js/test/test_base.js", + "test-base": "mocha --reporter spec js/test/test_base.js --reporter ololog/reporter", "export-exchanges": "node export-exchanges", "update-badges": "node update-badges", "convert-md-2-rst": "bash ./convert-md-2-rst", @@ -51,7 +51,7 @@ "istanbul": "^0.4.5", "mocha": "^3.5.3", "nyc": "^11.0.3", - "ololog": "^1.1.82" + "ololog": "^1.1.84" }, "author": { "name": "Igor Kroitor", From 1cb5a3f962baaf13bd509b0ddddfdd5cf9de266e Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 01:53:43 +0300 Subject: [PATCH 15/38] forgot to change a require path.. --- js/base/functions.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/js/base/functions.js b/js/base/functions.js index 9d0846128aead..8a2fb9ff79670 100644 --- a/js/base/functions.js +++ b/js/base/functions.js @@ -16,7 +16,7 @@ module.exports = unCamelCasePropertyNames ({ ...require ('./functions/platform'), ...require ('./functions/generic'), ...require ('./functions/string'), - ...require ('./functions/safe'), + ...require ('./functions/type'), ...require ('./functions/number'), ...require ('./functions/encode'), ...require ('./functions/crypto'), From 8815d1a7d57ee3c549e4058b672e1e684b5459bf Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 01:59:15 +0300 Subject: [PATCH 16/38] +isDictionary to type.js --- js/base/functions/generic.js | 4 ++++ js/base/functions/type.js | 10 ++++++---- js/test/test_base.js | 5 +++++ 3 files changed, 15 insertions(+), 4 deletions(-) diff --git a/js/base/functions/generic.js b/js/base/functions/generic.js index 38f8a651afd4b..4f04cb5922c29 100644 --- a/js/base/functions/generic.js +++ b/js/base/functions/generic.js @@ -2,6 +2,10 @@ /* ------------------------------------------------------------------------ */ +const { isObject, isNumber } = require ('./type') + +/* ------------------------------------------------------------------------ */ + const empty = () => Object.create (null) // empty obj without even a prototype , keys = Object.keys diff --git a/js/base/functions/type.js b/js/base/functions/type.js index 117da4cda7f3c..9857de2948c6d 100644 --- a/js/base/functions/type.js +++ b/js/base/functions/type.js @@ -2,9 +2,11 @@ /* ------------------------------------------------------------------------ */ -const isNumber = Number.isFinite - , isObject = o => typeof o === 'object' - , isString = s => typeof s === 'string' +const isNumber = Number.isFinite + , isArray = Array.isArray + , isObject = o => (typeof o === 'object') + , isDictionary = o => (typeof o === 'object') && !isArray (x) + , isString = s => (typeof s === 'string') , isStringCoercible = x => (hasProps (x) && x.toString) || isNumber (x) /* ............................................. */ @@ -39,7 +41,7 @@ module.exports = , safeInteger: (o, k, $default, n = asInteger (prop (o, k))) => isNumber (n) ? n : $default , safeValue: (o, k, $default, x = prop (o, k) ) => hasProps (x) ? x : $default , safeString: (o, k, $default, x = prop (o, k) ) => isStringCoercible (x) ? String (x) : $default - + } /* ------------------------------------------------------------------------ */ diff --git a/js/test/test_base.js b/js/test/test_base.js index fa7665ee3ef11..e37b0145906f3 100644 --- a/js/test/test_base.js +++ b/js/test/test_base.js @@ -415,6 +415,11 @@ describe ('ccxt base code', () => { assert.deepEqual (actual, expected) }) + + it.only ('sum works', () => { + + ccxt.sum ([1,20,300]).should.equal (321) + }) }) /* ------------------------------------------------------------------------ */ From 75ad08fb28587962908094e19e7b00bdc4cd0b8e Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 02:01:47 +0300 Subject: [PATCH 17/38] correct isObject implementation --- js/base/functions/type.js | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/js/base/functions/type.js b/js/base/functions/type.js index 9857de2948c6d..2439d7b3d30ce 100644 --- a/js/base/functions/type.js +++ b/js/base/functions/type.js @@ -4,9 +4,9 @@ const isNumber = Number.isFinite , isArray = Array.isArray - , isObject = o => (typeof o === 'object') - , isDictionary = o => (typeof o === 'object') && !isArray (x) - , isString = s => (typeof s === 'string') + , isString = s => (typeof s === 'string') + , isObject = o => (o !== null) && (typeof o === 'object') + , isDictionary = o => (isObject (o) && !isArray (x)) , isStringCoercible = x => (hasProps (x) && x.toString) || isNumber (x) /* ............................................. */ From 2b4ceb9a77ab0e55d08a89c0a88fe1cdccdcb49e Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 02:19:58 +0300 Subject: [PATCH 18/38] more tests for base functions, fixed tons of bugs & quirks --- js/base/functions/generic.js | 29 +++++++++++------------ js/base/functions/type.js | 4 +++- js/test/test_base.js | 45 ++++++++++++++++++++++++++++++++++-- 3 files changed, 60 insertions(+), 18 deletions(-) diff --git a/js/base/functions/generic.js b/js/base/functions/generic.js index 4f04cb5922c29..32fdd7928ab35 100644 --- a/js/base/functions/generic.js +++ b/js/base/functions/generic.js @@ -2,7 +2,7 @@ /* ------------------------------------------------------------------------ */ -const { isObject, isNumber } = require ('./type') +const { isObject, isNumber, isDictionary, isArray } = require ('./type') /* ------------------------------------------------------------------------ */ @@ -10,7 +10,7 @@ const empty = () => Object.create (null) // empty obj without even a prototype , keys = Object.keys - , values = x => !Array.isArray (x) // don't copy arrays if they're already arrays! + , values = x => !isArray (x) // don't copy arrays if they're already arrays! ? Object.values (x) : x @@ -18,9 +18,9 @@ const empty = () => Object.create (null) // empty obj without even a prototype , extend = (...args) => Object.assign (empty (), ...args) // NB: side-effect free - , clone = x => Array.isArray (x) - ? Array.from (x) // clones arrays - : extend (x) // clones objects + , clone = x => isArray (x) + ? Array.from (x) // clones arrays + : extend (x) // clones objects /* ------------------------------------------------------------------------ */ @@ -95,7 +95,7 @@ module.exports = { , flatten: function flatten (x, out = []) { for (const v of x) { - if (Array.isArray (v)) flatten (v, out) + if (isArray (v)) flatten (v, out) else out.push (v) } @@ -116,22 +116,21 @@ module.exports = { for (const k of args) { - if (typeof k === 'string') // omit (x, 'a', 'b') - delete out[k] - - else if (Array.isArray (k)) // omit (x, ['a', 'b']) + if (isArray (k)) // omit (x, ['a', 'b']) for (const kk of k) delete out[kk] + + else delete out[k] // omit (x, 'a', 'b') } return out } - + /* ............................................. */ , sum (...xs) { - const ns = xs.filter (Number.isFinite) // leave only numbers + const ns = xs.filter (isNumber) // leave only numbers return (ns.length > 0) ? ns.reduce ((a, b) => a + b, 0) @@ -146,10 +145,10 @@ module.exports = { for (const x of xs) { - if (x && (typeof x === 'object') && !Array.isArray (x)) { + if (isDictionary (x)) { - if (typeof out !== 'object') - out = Object.create (null) + if (!isObject (out)) + out = empty () for (const k in x) out[k] = deepExtend (out[k], x[k]) diff --git a/js/base/functions/type.js b/js/base/functions/type.js index 2439d7b3d30ce..555d06be129e6 100644 --- a/js/base/functions/type.js +++ b/js/base/functions/type.js @@ -6,7 +6,7 @@ const isNumber = Number.isFinite , isArray = Array.isArray , isString = s => (typeof s === 'string') , isObject = o => (o !== null) && (typeof o === 'object') - , isDictionary = o => (isObject (o) && !isArray (x)) + , isDictionary = o => (isObject (o) && !isArray (o)) , isStringCoercible = x => (hasProps (x) && x.toString) || isNumber (x) /* ............................................. */ @@ -27,9 +27,11 @@ const asFloat = x => (isNumber (x) || isString (x)) ? parseFloat (x) : NaN module.exports = { isNumber + , isArray , isObject , isString , isStringCoercible + , isDictionary , hasProps , prop diff --git a/js/test/test_base.js b/js/test/test_base.js index e37b0145906f3..f0a56770558c4 100644 --- a/js/test/test_base.js +++ b/js/test/test_base.js @@ -416,9 +416,50 @@ describe ('ccxt base code', () => { assert.deepEqual (actual, expected) }) - it.only ('sum works', () => { + it ('omit works', () => { + + assert.deepEqual (ccxt.omit ({ }, 'foo'), {}) + assert.deepEqual (ccxt.omit ({ foo: 2 }, 'foo'), { }) + assert.deepEqual (ccxt.omit ({ foo: 2, bar: 3 }, 'foo'), { bar: 3 }) + assert.deepEqual (ccxt.omit ({ foo: 2, bar: 3 }, ['foo']), { bar: 3 }) + assert.deepEqual (ccxt.omit ({ foo: 2, bar: 3 }), { foo: 2, bar: 3 }) + assert.deepEqual (ccxt.omit ({ foo: 2, bar: 3 }, 'foo', 'bar'), {}) + assert.deepEqual (ccxt.omit ({ foo: 2, bar: 3 }, ['foo'], 'bar'), {}) + assert.deepEqual (ccxt.omit ({ 5: 2, bar: 3 }, [5]), { bar: 3 }) + assert.deepEqual (ccxt.omit ({ 5: 2, bar: 3 }, 5), { bar: 3 }) + }) + + it ('sum works', () => { + + assert (ccxt.sum () === undefined) + + ccxt.sum (2).should.equal (2) + ccxt.sum (2,30,400).should.equal (432) + ccxt.sum (2,undefined,[88],30,'7',400,null).should.equal (432) + }) - ccxt.sum ([1,20,300]).should.equal (321) + it ('sortBy works', () => { + + const arr = [{ x: 5 }, { x: 2 }, { x: 4 }, { x: 0 },{ x: 1 },{ x: 3 }] + ccxt.sortBy (arr, 'x') + + assert.deepEqual (arr + [ { x: 0 }, + { x: 1 }, + { x: 2 }, + { x: 3 }, + { x: 4 }, + { x: 5 } ]) + + assert.deepEqual (ccxt.sortBy (arr, 'x', true), + [ { x: 5 }, + { x: 4 }, + { x: 3 }, + { x: 2 }, + { x: 1 }, + { x: 0 } ]) + + assert.deepEqual (ccxt.sortBy ([], 'x'), []) }) }) From dcc40b6ec6d6c912e13feebb74678b2a30769f71 Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 02:30:32 +0300 Subject: [PATCH 19/38] more fixes, now basic exchange test passes --- js/base/Exchange.js | 10 ++++++---- js/base/functions/crypto.js | 2 +- js/base/functions/throttle.js | 2 +- run-tests.js | 2 +- 4 files changed, 9 insertions(+), 7 deletions(-) diff --git a/js/base/Exchange.js b/js/base/Exchange.js index 469bcd3156ddb..794ca620586f1 100644 --- a/js/base/Exchange.js +++ b/js/base/Exchange.js @@ -8,8 +8,6 @@ const functions = require ('./functions') const { isNode , deepExtend , extend - , sleep - , timeout , flatten , indexBy , sortBy @@ -17,8 +15,12 @@ const { isNode , aggregate , uuid , precisionFromString - , throttle - , time } = functions + , throttle } = functions + +const { sleep + , timeout + , time + , TimedOut } = require ('./functions/time') const { ExchangeError , NotSupported diff --git a/js/base/functions/crypto.js b/js/base/functions/crypto.js index 7c93adeb9deda..535e8b92ef3fc 100644 --- a/js/base/functions/crypto.js +++ b/js/base/functions/crypto.js @@ -3,7 +3,7 @@ /* ------------------------------------------------------------------------ */ const CryptoJS = require ('crypto-js') -const { capitalize } = require ('./generic') +const { capitalize } = require ('./string') const { stringToBase64, utf16ToBase64, urlencodeBase64 } = require ('./encode') /* ------------------------------------------------------------------------ */ diff --git a/js/base/functions/throttle.js b/js/base/functions/throttle.js index 25169a3a357c1..417e66e40b749 100644 --- a/js/base/functions/throttle.js +++ b/js/base/functions/throttle.js @@ -34,7 +34,7 @@ module.exports = { const hasEnoughTokens = cfg.capacity ? (numTokens > 0) : (numTokens >= 0) if (hasEnoughTokens) { if (queue.length > 0) { - const { cost, resolve, reject } = queue[0] + let { cost, resolve, reject } = queue[0] cost = (cost || cfg.defaultCost) if (numTokens >= Math.min (cost, cfg.capacity)) { numTokens -= cost diff --git a/run-tests.js b/run-tests.js index af91c02e2a6cf..ff9ecb95fe2d6 100644 --- a/run-tests.js +++ b/run-tests.js @@ -125,7 +125,7 @@ const sequentialMap = async (input, fn) => { const testExchange = async (exchange) => { - const nonce = Date.now () + const nonce = ccxt.time.now () /* Run tests for all/selected languages (in parallel) */ From 8954a4a300baac69846e038cd3de8fa63a90b01e Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 05:45:14 +0300 Subject: [PATCH 20/38] =?UTF-8?q?better=20camelCase=20conversion=20regexp?= =?UTF-8?q?=20+=20test=20(handles=20hasFetchOHLCV=20=E2=86=92=20has=5Ffetc?= =?UTF-8?q?h=5FOHLCV=20conversion)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- js/base/Exchange.js | 17 ++++++++++++----- js/base/functions/string.js | 3 ++- js/test/test_base.js | 31 +++++++++++++++++++++++++++---- 3 files changed, 41 insertions(+), 10 deletions(-) diff --git a/js/base/Exchange.js b/js/base/Exchange.js index 794ca620586f1..177e140889fe0 100644 --- a/js/base/Exchange.js +++ b/js/base/Exchange.js @@ -6,6 +6,8 @@ const functions = require ('./functions') , Market = require ('./Market') const { isNode + , keys + , values , deepExtend , extend , flatten @@ -14,6 +16,7 @@ const { isNode , groupBy , aggregate , uuid + , unCamelCase , precisionFromString , throttle } = functions @@ -154,6 +157,11 @@ module.exports = class Exchange { this.arrayConcat = (a, b) => a.concat (b) // TODO: generate + + for (const k of keys (this)) { + this[unCamelCase (k)] = this[k] + } + this.market_id = this.marketId this.market_ids = this.marketIds this.array_concat = this.arrayConcat @@ -201,11 +209,10 @@ module.exports = class Exchange { this.amount_to_string = this.amountToString this.fee_to_precision = this.feeToPrecision this.cost_to_precision = this.costToPrecision - this.precisionFromString = precisionFromString - this.precision_from_string = precisionFromString - this.truncate = functions.truncate - this.truncate_to_string = functions.truncate_to_string - this.uuid = uuid + this.precision_from_string = this.precisionFromString + this.truncate = this.truncate + this.truncate_to_string = this.truncateToString + this.uuid = this.uuid // API methods metainfo this.has = { diff --git a/js/base/functions/string.js b/js/base/functions/string.js index d29acfb885329..1a7530bf50bf8 100644 --- a/js/base/functions/string.js +++ b/js/base/functions/string.js @@ -7,7 +7,8 @@ module.exports = { uuid: a => a ? (a ^ Math.random () * 16 >> a / 4).toString (16) : ([1e7]+-1e3+-4e3+-8e3+-1e11).replace (/[018]/g, uuid) - , unCamelCase: s => s.replace (/[a-z][A-Z]/g, x => x[0] + '_' + x[1].toLowerCase ()) // fromCamelCase → from_camel_case + , unCamelCase: s => s.replace (/[a-z][A-Z]/g, x => x[0] + '_' + x[1].toLowerCase ()) // hasFetchOHLCV → has_fetch_oHLCV + .replace (/[a-z][A-Z]/g, x => x[0].toUpperCase () + x[1]) // has_fetch_oHLCV → has_fetch_OHLCV , capitalize: s => s.length ? (s.charAt (0).toUpperCase () + s.slice (1)) diff --git a/js/test/test_base.js b/js/test/test_base.js index f0a56770558c4..9a2649009514f 100644 --- a/js/test/test_base.js +++ b/js/test/test_base.js @@ -7,6 +7,10 @@ global.log = require ('ololog') // for easier debugging const ccxt = require ('../../ccxt.js') , assert = require ('assert') , ansi = require ('ansicolor').nice; + +/* ------------------------------------------------------------------------ */ + +const { keys, values, unique, index } = ccxt /* ------------------------------------------------------------------------ */ @@ -33,7 +37,7 @@ describe ('ccxt base code', () => { }) - it ('safeFloat is robust', async () => { + it ('safeFloat/safeInteger is robust', async () => { const $default = {} @@ -68,11 +72,12 @@ describe ('ccxt base code', () => { it ('setTimeout_safe is working', (done) => { - const start = Date.now () + const time = ccxt.time + const start = time.now () const calls = [] const brokenSetTimeout = (done, ms) => { - calls.push ({ when: Date.now () - start, ms_asked: ms }) + calls.push ({ when: time.now () - start, ms_asked: ms }) return setTimeout (done, 100) // simulates a defect setTimeout implementation that sleeps wrong time (100ms always in this test) } @@ -182,7 +187,7 @@ describe ('ccxt base code', () => { tokenBucket: { capacity, numTokens, defaultCost, delay }, async ping (...args) { return this.throttle ().then (() => exchange.pong (...args)) }, - async pong (...args) { calls.push ({ when: Date.now (), path: args[0], args }) } + async pong (...args) { calls.push ({ when: ccxt.time.now (), path: args[0], args }) } }) await exchange.ping ('foo') @@ -461,6 +466,24 @@ describe ('ccxt base code', () => { assert.deepEqual (ccxt.sortBy ([], 'x'), []) }) + + it ('camelCase/camel_case property conversion works', () => { + + const exchange = new ccxt.Exchange ({ 'id': 'mock' }) + + const propsSeenBefore = index (["isNode", "empty", "keys", "values", "extend", "clone", "index", "ordered", "unique", "keysort", "indexBy", "groupBy", "filterBy", "sortBy", "flatten", "pluck", "omit", "sum", "deepExtend", "uuid", "unCamelCase", "capitalize", "isNumber", "isArray", "isObject", "isString", "isStringCoercible", "isDictionary", "hasProps", "prop", "asFloat", "asInteger", "safeFloat", "safeInteger", "safeValue", "safeString", "decimal", "toFixed", "truncate", "truncateToString", "precisionFromString", "stringToBinary", "stringToBase64", "utf16ToBase64", "base64ToBinary", "base64ToString", "binaryToString", "binaryConcat", "urlencode", "rawencode", "urlencodeBase64", "hash", "hmac", "jwt", "time", "setTimeout_safe", "sleep", "TimedOut", "timeout", "throttle", "json", "unjson", "aggregate", "is_node", "index_by", "group_by", "filter_by", "sort_by", "deep_extend", "un_camel_case", "is_number", "is_array", "is_object", "is_string", "is_string_coercible", "is_dictionary", "has_props", "as_float", "as_integer", "safe_float", "safe_integer", "safe_value", "safe_string", "to_fixed", "truncate_to_string", "precision_from_string", "string_to_binary", "string_to_base64", "utf16To_base64", "base64To_binary", "base64To_string", "binary_to_string", "binary_concat", "urlencode_base64", "set_timeout_safe", "Timed_out", "encode", "decode", "nodeVersion", "userAgents", "headers", "proxy", "origin", "iso8601", "parse8601", "milliseconds", "microseconds", "seconds", "id", "enableRateLimit", "rateLimit", "parseJsonResponse", "substituteCommonCurrencyCodes", "parseBalanceFromOpenOrders", "verbose", "debug", "journal", "userAgent", "twofa", "timeframes", "hasPublicAPI", "hasPrivateAPI", "hasCORS", "hasDeposit", "hasFetchBalance", "hasFetchClosedOrders", "hasFetchCurrencies", "hasFetchMyTrades", "hasFetchOHLCV", "hasFetchOpenOrders", "hasFetchOrder", "hasFetchOrderBook", "hasFetchOrders", "hasFetchTicker", "hasFetchTickers", "hasFetchBidsAsks", "hasFetchTrades", "hasWithdraw", "hasCreateOrder", "hasCancelOrder", "apiKey", "secret", "uid", "login", "password", "requiredCredentials", "exceptions", "balance", "orderbooks", "tickers", "fees", "orders", "trades", "currencies", "last_http_response", "last_json_response", "arrayConcat", "market_id", "market_ids", "array_concat", "implode_params", "extract_params", "fetch_balance", "fetch_free_balance", "fetch_used_balance","fetch_total_balance", "fetch_l2_order_book", "fetch_order_book", "fetch_bids_asks", "fetch_tickers", "fetch_ticker", "fetch_trades", "fetch_order", "fetch_orders", "fetch_open_orders", "fetch_closed_orders", "fetch_order_status", "fetch_markets", "load_markets", "set_markets", "parse_balance", "parse_bid_ask", "parse_bids_asks", "parse_order_book", "parse_trades", "parse_orders", "parse_ohlcv", "parse_ohlcvs", "edit_limit_buy_order", "edit_limit_sell_order", "edit_limit_order", "edit_order", "create_limit_buy_order", "create_limit_sell_order", "create_market_buy_order", "create_market_sell_order", "create_order", "calculate_fee", "common_currency_code", "price_to_precision", "amount_to_precision", "amount_to_string", "fee_to_precision", "cost_to_precision", "has", "tokenBucket", "executeRestRequest"]) + const props = index (keys (exchange)) + + + for (const k of Array.from (propsSeenBefore)) + if (!props.has (k)) + throw new Error (`missing prop: ${k}`) + + for (const k of Array.from (props)) + if (!propsSeenBefore.has (k)) + log.magenta.noLocate (`+ ${k}`) + + }) }) /* ------------------------------------------------------------------------ */ From 751c9ca605163b365c4dcc6524d16224f62cd588 Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 05:56:18 +0300 Subject: [PATCH 21/38] property names with underscores now automatically generated from camelCase counterparts in Exchange constructor (greatly debloated the constructor code) --- js/base/Exchange.js | 60 ++++---------------------------------------- js/test/test_base.js | 13 ++++++---- 2 files changed, 13 insertions(+), 60 deletions(-) diff --git a/js/base/Exchange.js b/js/base/Exchange.js index 177e140889fe0..4b812222d8517 100644 --- a/js/base/Exchange.js +++ b/js/base/Exchange.js @@ -158,62 +158,12 @@ module.exports = class Exchange { // TODO: generate - for (const k of keys (this)) { - this[unCamelCase (k)] = this[k] - } - - this.market_id = this.marketId - this.market_ids = this.marketIds - this.array_concat = this.arrayConcat - this.implode_params = this.implodeParams - this.extract_params = this.extractParams - this.fetch_balance = this.fetchBalance - this.fetch_free_balance = this.fetchFreeBalance - this.fetch_used_balance = this.fetchUsedBalance - this.fetch_total_balance = this.fetchTotalBalance - this.fetch_l2_order_book = this.fetchL2OrderBook - this.fetch_order_book = this.fetchOrderBook - this.fetch_bids_asks = this.fetchBidsAsks - this.fetch_tickers = this.fetchTickers - this.fetch_ticker = this.fetchTicker - this.fetch_trades = this.fetchTrades - this.fetch_order = this.fetchOrder - this.fetch_orders = this.fetchOrders - this.fetch_open_orders = this.fetchOpenOrders - this.fetch_closed_orders = this.fetchClosedOrders - this.fetch_order_status = this.fetchOrderStatus - this.fetch_markets = this.fetchMarkets - this.load_markets = this.loadMarkets - this.set_markets = this.setMarkets - this.parse_balance = this.parseBalance - this.parse_bid_ask = this.parseBidAsk - this.parse_bids_asks = this.parseBidsAsks - this.parse_order_book = this.parseOrderBook - this.parse_trades = this.parseTrades - this.parse_orders = this.parseOrders - this.parse_ohlcv = this.parseOHLCV - this.parse_ohlcvs = this.parseOHLCVs - this.edit_limit_buy_order = this.editLimitBuyOrder - this.edit_limit_sell_order = this.editLimitSellOrder - this.edit_limit_order = this.editLimitOrder - this.edit_order = this.editOrder - this.create_limit_buy_order = this.createLimitBuyOrder - this.create_limit_sell_order = this.createLimitSellOrder - this.create_market_buy_order = this.createMarketBuyOrder - this.create_market_sell_order = this.createMarketSellOrder - this.create_order = this.createOrder - this.calculate_fee = this.calculateFee - this.common_currency_code = this.commonCurrencyCode - this.price_to_precision = this.priceToPrecision - this.amount_to_precision = this.amountToPrecision - this.amount_to_string = this.amountToString - this.fee_to_precision = this.feeToPrecision - this.cost_to_precision = this.costToPrecision - this.precision_from_string = this.precisionFromString - this.truncate = this.truncate - this.truncate_to_string = this.truncateToString - this.uuid = this.uuid + const names = Object.getOwnPropertyNames (this).concat ( + Object.getOwnPropertyNames (this.constructor.prototype)) + for (const k of names) + this[unCamelCase (k)] = this[k] + // API methods metainfo this.has = { 'cancelOrder': this.hasPrivateAPI, diff --git a/js/test/test_base.js b/js/test/test_base.js index 9a2649009514f..9c39f1350e8d3 100644 --- a/js/test/test_base.js +++ b/js/test/test_base.js @@ -467,16 +467,19 @@ describe ('ccxt base code', () => { assert.deepEqual (ccxt.sortBy ([], 'x'), []) }) - it ('camelCase/camel_case property conversion works', () => { + it.only ('camelCase/camel_case property conversion works', () => { const exchange = new ccxt.Exchange ({ 'id': 'mock' }) - const propsSeenBefore = index (["isNode", "empty", "keys", "values", "extend", "clone", "index", "ordered", "unique", "keysort", "indexBy", "groupBy", "filterBy", "sortBy", "flatten", "pluck", "omit", "sum", "deepExtend", "uuid", "unCamelCase", "capitalize", "isNumber", "isArray", "isObject", "isString", "isStringCoercible", "isDictionary", "hasProps", "prop", "asFloat", "asInteger", "safeFloat", "safeInteger", "safeValue", "safeString", "decimal", "toFixed", "truncate", "truncateToString", "precisionFromString", "stringToBinary", "stringToBase64", "utf16ToBase64", "base64ToBinary", "base64ToString", "binaryToString", "binaryConcat", "urlencode", "rawencode", "urlencodeBase64", "hash", "hmac", "jwt", "time", "setTimeout_safe", "sleep", "TimedOut", "timeout", "throttle", "json", "unjson", "aggregate", "is_node", "index_by", "group_by", "filter_by", "sort_by", "deep_extend", "un_camel_case", "is_number", "is_array", "is_object", "is_string", "is_string_coercible", "is_dictionary", "has_props", "as_float", "as_integer", "safe_float", "safe_integer", "safe_value", "safe_string", "to_fixed", "truncate_to_string", "precision_from_string", "string_to_binary", "string_to_base64", "utf16To_base64", "base64To_binary", "base64To_string", "binary_to_string", "binary_concat", "urlencode_base64", "set_timeout_safe", "Timed_out", "encode", "decode", "nodeVersion", "userAgents", "headers", "proxy", "origin", "iso8601", "parse8601", "milliseconds", "microseconds", "seconds", "id", "enableRateLimit", "rateLimit", "parseJsonResponse", "substituteCommonCurrencyCodes", "parseBalanceFromOpenOrders", "verbose", "debug", "journal", "userAgent", "twofa", "timeframes", "hasPublicAPI", "hasPrivateAPI", "hasCORS", "hasDeposit", "hasFetchBalance", "hasFetchClosedOrders", "hasFetchCurrencies", "hasFetchMyTrades", "hasFetchOHLCV", "hasFetchOpenOrders", "hasFetchOrder", "hasFetchOrderBook", "hasFetchOrders", "hasFetchTicker", "hasFetchTickers", "hasFetchBidsAsks", "hasFetchTrades", "hasWithdraw", "hasCreateOrder", "hasCancelOrder", "apiKey", "secret", "uid", "login", "password", "requiredCredentials", "exceptions", "balance", "orderbooks", "tickers", "fees", "orders", "trades", "currencies", "last_http_response", "last_json_response", "arrayConcat", "market_id", "market_ids", "array_concat", "implode_params", "extract_params", "fetch_balance", "fetch_free_balance", "fetch_used_balance","fetch_total_balance", "fetch_l2_order_book", "fetch_order_book", "fetch_bids_asks", "fetch_tickers", "fetch_ticker", "fetch_trades", "fetch_order", "fetch_orders", "fetch_open_orders", "fetch_closed_orders", "fetch_order_status", "fetch_markets", "load_markets", "set_markets", "parse_balance", "parse_bid_ask", "parse_bids_asks", "parse_order_book", "parse_trades", "parse_orders", "parse_ohlcv", "parse_ohlcvs", "edit_limit_buy_order", "edit_limit_sell_order", "edit_limit_order", "edit_order", "create_limit_buy_order", "create_limit_sell_order", "create_market_buy_order", "create_market_sell_order", "create_order", "calculate_fee", "common_currency_code", "price_to_precision", "amount_to_precision", "amount_to_string", "fee_to_precision", "cost_to_precision", "has", "tokenBucket", "executeRestRequest"]) - const props = index (keys (exchange)) - + const propsSeenBefore = index ( + ["isNode", "empty", "keys", "values", "extend", "clone", "index", "ordered", "unique", "keysort", "indexBy", "groupBy", "filterBy", "sortBy", "flatten", "pluck", "omit", "sum", "deepExtend", "uuid", "unCamelCase", "capitalize", "isNumber", "isArray", "isObject", "isString", "isStringCoercible", "isDictionary", "hasProps", "prop", "asFloat", "asInteger", "safeFloat", "safeInteger", "safeValue", "safeString", "decimal", "toFixed", "truncate", "truncateToString", "precisionFromString", "stringToBinary", "stringToBase64", "utf16ToBase64", "base64ToBinary", "base64ToString", "binaryToString", "binaryConcat", "urlencode", "rawencode", "urlencodeBase64", "hash", "hmac", "jwt", "time", "setTimeout_safe", "sleep", "TimedOut", "timeout", "throttle", "json", "unjson", "aggregate", "is_node", "index_by", "group_by", "filter_by", "sort_by", "deep_extend", "un_camel_case", "is_number", "is_array", "is_object", "is_string", "is_string_coercible", "is_dictionary", "has_props", "as_float", "as_integer", "safe_float", "safe_integer", "safe_value", "safe_string", "to_fixed", "truncate_to_string", "precision_from_string", "string_to_binary", "string_to_base64", "utf16To_base64", "base64To_binary", "base64To_string", "binary_to_string", "binary_concat", "urlencode_base64", "set_timeout_safe", "Timed_out", "encode", "decode", "nodeVersion", "userAgents", "headers", "proxy", "origin", "iso8601", "parse8601", "milliseconds", "microseconds", "seconds", "id", "enableRateLimit", "rateLimit", "parseJsonResponse", "substituteCommonCurrencyCodes", "parseBalanceFromOpenOrders", "verbose", "debug", "journal", "userAgent", "twofa", "timeframes", "hasPublicAPI", "hasPrivateAPI", "hasCORS", "hasDeposit", "hasFetchBalance", "hasFetchClosedOrders", "hasFetchCurrencies", "hasFetchMyTrades", "hasFetchOHLCV", "hasFetchOpenOrders", "hasFetchOrder", "hasFetchOrderBook", "hasFetchOrders", "hasFetchTicker", "hasFetchTickers", "hasFetchBidsAsks", "hasFetchTrades", "hasWithdraw", "hasCreateOrder", "hasCancelOrder", "apiKey", "secret", "uid", "login", "password", "requiredCredentials", "exceptions", "balance", "orderbooks", "tickers", "fees", "orders", "trades", "currencies", "last_http_response", "last_json_response", "arrayConcat", "market_id", "market_ids", "array_concat", "implode_params", "extract_params", "fetch_balance", "fetch_free_balance", "fetch_used_balance","fetch_total_balance", "fetch_l2_order_book", "fetch_order_book", "fetch_bids_asks", "fetch_tickers", "fetch_ticker", "fetch_trades", "fetch_order", "fetch_orders", "fetch_open_orders", "fetch_closed_orders", "fetch_order_status", "fetch_markets", "load_markets", "set_markets", "parse_balance", "parse_bid_ask", "parse_bids_asks", "parse_order_book", "parse_trades", "parse_orders", "parse_ohlcv", "parse_ohlcvs", "edit_limit_buy_order", "edit_limit_sell_order", "edit_limit_order", "edit_order", "create_limit_buy_order", "create_limit_sell_order", "create_market_buy_order", "create_market_sell_order", "create_order", "calculate_fee", "common_currency_code", "price_to_precision", "amount_to_precision", "amount_to_string", "fee_to_precision", "cost_to_precision", "constructor", "getMarket", "describe", "defaults", "nonce", "encodeURIComponent", "checkRequiredCredentials", "initRestRateLimiter", "defineRestApi", "fetch", "fetch2", "request", "handleErrors", "defaultErrorHandler", "handleRestErrors", "handleRestResponse", "setMarkets", "loadMarkets", "fetchBidsAsks", "fetchTickers", "fetchOrder", "fetchOrders", "fetchOpenOrders", "fetchClosedOrders", "fetchMyTrades", "fetchCurrencies", "fetchMarkets", "fetchOrderStatus", "account", "commonCurrencyCode", "currency", "market", "marketId", "marketIds", "symbol", "extractParams", "implodeParams", "url", "parseBidAsk", "parseBidsAsks", "fetchL2OrderBook", "parseOrderBook", "getCurrencyUsedOnOpenOrders", "parseBalance", "fetchPartialBalance", "fetchFreeBalance", "fetchUsedBalance", "fetchTotalBalance", "filterBySinceLimit", "parseTrades", "parseOrders", "filterOrdersBySymbol", "parseOHLCV", "parseOHLCVs", "editLimitBuyOrder", "editLimitSellOrder", "editLimitOrder", "editOrder", "createLimitBuyOrder", "createLimitSellOrder", "createMarketBuyOrder", "createMarketSellOrder", "costToPrecision", "priceToPrecision", "amountToPrecision", "amountToString", "amountToLots", "feeToPrecision", "calculateFee", "Ymd", "YmdHMS"] + ) + + const props = index (Object.getOwnPropertyNames (exchange).concat ( + Object.getOwnPropertyNames (exchange.constructor.prototype))) for (const k of Array.from (propsSeenBefore)) - if (!props.has (k)) + if (this[k] && !props.has (k)) throw new Error (`missing prop: ${k}`) for (const k of Array.from (props)) From 0b2fafe475262b81e411003b2df0eb867dea5225 Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 06:03:10 +0300 Subject: [PATCH 22/38] =?UTF-8?q?has=5Ffetch=5FOHLCV=20=E2=86=92=20has=5Ff?= =?UTF-8?q?etch=5Fohlcv=20/=20fetchL2OrderBook=20=E2=86=92=20fetch=5Fl2=5F?= =?UTF-8?q?order=5Fbook=20(automatic=20conversion)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- js/base/Exchange.js | 4 ++-- js/base/functions/string.js | 3 +-- 2 files changed, 3 insertions(+), 4 deletions(-) diff --git a/js/base/Exchange.js b/js/base/Exchange.js index 4b812222d8517..2783346fc9e8b 100644 --- a/js/base/Exchange.js +++ b/js/base/Exchange.js @@ -107,6 +107,8 @@ module.exports = class Exchange { this.userAgent = undefined this.twofa = false // two-factor authentication (2FA) this.timeframes = undefined + + // TODO: generate this.hasPublicAPI = true this.hasPrivateAPI = true this.hasCORS = false @@ -156,8 +158,6 @@ module.exports = class Exchange { this.arrayConcat = (a, b) => a.concat (b) - // TODO: generate - const names = Object.getOwnPropertyNames (this).concat ( Object.getOwnPropertyNames (this.constructor.prototype)) diff --git a/js/base/functions/string.js b/js/base/functions/string.js index 1a7530bf50bf8..db9b8090a0df7 100644 --- a/js/base/functions/string.js +++ b/js/base/functions/string.js @@ -7,8 +7,7 @@ module.exports = { uuid: a => a ? (a ^ Math.random () * 16 >> a / 4).toString (16) : ([1e7]+-1e3+-4e3+-8e3+-1e11).replace (/[018]/g, uuid) - , unCamelCase: s => s.replace (/[a-z][A-Z]/g, x => x[0] + '_' + x[1].toLowerCase ()) // hasFetchOHLCV → has_fetch_oHLCV - .replace (/[a-z][A-Z]/g, x => x[0].toUpperCase () + x[1]) // has_fetch_oHLCV → has_fetch_OHLCV + , unCamelCase: s => s.replace (/[a-z0-9][A-Z]/g, x => x[0] + '_' + x[1]).toLowerCase () // hasFetchOHLCV → has_fetch_ohlcv , capitalize: s => s.length ? (s.charAt (0).toUpperCase () + s.slice (1)) From 9e5ed23d44277d34887f97f66c298c0b4e332afc Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 06:05:45 +0300 Subject: [PATCH 23/38] removed obsolete metainfo interface in coinexchange.js --- js/coinexchange.js | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/js/coinexchange.js b/js/coinexchange.js index 347e165d90f49..ff37833961f98 100644 --- a/js/coinexchange.js +++ b/js/coinexchange.js @@ -15,13 +15,9 @@ module.exports = class coinexchange extends Exchange { 'name': 'CoinExchange', 'countries': [ 'IN', 'JP', 'KR', 'VN', 'US' ], 'rateLimit': 1000, - // obsolete metainfo interface - 'hasPrivateAPI': false, - 'hasFetchTrades': false, - 'hasFetchCurrencies': true, - 'hasFetchTickers': true, // new metainfo interface 'has': { + 'privateAPI': false, 'fetchTrades': false, 'fetchCurrencies': true, 'fetchTickers': true, From f26372c730844ebb10a951dae89ab54561e36d91 Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 06:06:39 +0300 Subject: [PATCH 24/38] removed obsolete metainfo interface in coinmarketcap.js --- js/coinmarketcap.js | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/js/coinmarketcap.js b/js/coinmarketcap.js index cb99128c05fb8..65f3bd9418b1c 100644 --- a/js/coinmarketcap.js +++ b/js/coinmarketcap.js @@ -16,16 +16,16 @@ module.exports = class coinmarketcap extends Exchange { 'rateLimit': 10000, 'version': 'v1', 'countries': 'US', - 'hasCORS': true, - 'hasPrivateAPI': false, - 'hasCreateOrder': false, - 'hasCancelOrder': false, - 'hasFetchBalance': false, - 'hasFetchOrderBook': false, - 'hasFetchTrades': false, - 'hasFetchTickers': true, - 'hasFetchCurrencies': true, 'has': { + 'CORS': true, + 'privateAPI': false, + 'createOrder': false, + 'cancelOrder': false, + 'fetchBalance': false, + 'fetchOrderBook': false, + 'fetchTrades': false, + 'fetchTickers': true, + 'fetchCurrencies': true, 'fetchCurrencies': true, }, 'urls': { From 6cdbf2420cb8891e967d25e114b15d236f2d67e0 Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 06:08:25 +0300 Subject: [PATCH 25/38] removed obsolete metainfo interface in bibox.js --- js/bibox.js | 11 ++--------- 1 file changed, 2 insertions(+), 9 deletions(-) diff --git a/js/bibox.js b/js/bibox.js index baa6e38a407c0..7c679ec471653 100644 --- a/js/bibox.js +++ b/js/bibox.js @@ -15,16 +15,9 @@ module.exports = class bibox extends Exchange { 'name': 'Bibox', 'countries': [ 'CN', 'US', 'KR' ], 'version': 'v1', - 'hasCORS': false, - 'hasPublicAPI': false, - 'hasFetchBalance': true, - 'hasFetchCurrencies': true, - 'hasFetchTickers': true, - 'hasFetchOrders': true, - 'hasFetchMyTrades': true, - 'hasFetchOHLCV': true, - 'hasWithdraw': true, 'has': { + 'CORS': false, + 'publicAPI': false, 'fetchBalance': true, 'fetchCurrencies': true, 'fetchTickers': true, From bbf8641aa9cc22a5ce1ab879b82c41478041c729 Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 06:08:59 +0300 Subject: [PATCH 26/38] removed obsolete metainfo interface in binance.js --- js/binance.js | 11 +---------- 1 file changed, 1 insertion(+), 10 deletions(-) diff --git a/js/binance.js b/js/binance.js index f120766bab9fd..22b3afc123d34 100644 --- a/js/binance.js +++ b/js/binance.js @@ -15,18 +15,9 @@ module.exports = class binance extends Exchange { 'name': 'Binance', 'countries': 'JP', // Japan 'rateLimit': 500, - 'hasCORS': false, - // obsolete metainfo interface - 'hasFetchBidsAsks': true, - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, - 'hasFetchMyTrades': true, - 'hasFetchOrder': true, - 'hasFetchOrders': true, - 'hasFetchOpenOrders': true, - 'hasWithdraw': true, // new metainfo interface 'has': { + 'CORS': false, 'fetchBidsAsks': true, 'fetchTickers': true, 'fetchOHLCV': true, From fa6ad460ec75f7b71c430522b1c269b7f9a4ac9a Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 06:09:27 +0300 Subject: [PATCH 27/38] removed obsolete metainfo interface in bitcoincoid.js --- js/bitcoincoid.js | 10 ---------- 1 file changed, 10 deletions(-) diff --git a/js/bitcoincoid.js b/js/bitcoincoid.js index 0bd5650cab3e2..051df4a8eb41a 100644 --- a/js/bitcoincoid.js +++ b/js/bitcoincoid.js @@ -15,16 +15,6 @@ module.exports = class bitcoincoid extends Exchange { 'name': 'Bitcoin.co.id', 'countries': 'ID', // Indonesia 'hasCORS': false, - // obsolete metainfo interface - 'hasFetchTickers': false, - 'hasFetchOHLCV': false, - 'hasFetchOrder': true, - 'hasFetchOrders': false, - 'hasFetchClosedOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchMyTrades': false, - 'hasFetchCurrencies': false, - 'hasWithdraw': false, // new metainfo interface 'has': { 'fetchTickers': false, From 0fbf42311a6dd12244bd3b78f8b6fa42aa6d3b82 Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 06:10:10 +0300 Subject: [PATCH 28/38] removed obsolete metainfo interface in bitfinex.js --- js/bitfinex.js | 10 +--------- 1 file changed, 1 insertion(+), 9 deletions(-) diff --git a/js/bitfinex.js b/js/bitfinex.js index 1e1ce7dc7cdda..7559951e22bfd 100644 --- a/js/bitfinex.js +++ b/js/bitfinex.js @@ -16,17 +16,9 @@ module.exports = class bitfinex extends Exchange { 'countries': 'VG', 'version': 'v1', 'rateLimit': 1500, - 'hasCORS': false, - // old metainfo interface - 'hasFetchOrder': true, - 'hasFetchTickers': true, - 'hasDeposit': true, - 'hasWithdraw': true, - 'hasFetchOHLCV': true, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, // new metainfo interface 'has': { + 'CORS': false, 'fetchOHLCV': true, 'fetchTickers': true, 'fetchOrder': true, From b2ab52d513e1a274ff66cd05b183c9f37ebc5aa8 Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 06:21:25 +0300 Subject: [PATCH 29/38] removed obsolete metainfo interface in poloniex.js --- js/poloniex.js | 10 ---------- 1 file changed, 10 deletions(-) diff --git a/js/poloniex.js b/js/poloniex.js index b8200f586b2a8..fbf1abb72c698 100644 --- a/js/poloniex.js +++ b/js/poloniex.js @@ -16,16 +16,6 @@ module.exports = class poloniex extends Exchange { 'countries': 'US', 'rateLimit': 1000, // up to 6 calls per second 'hasCORS': true, - // obsolete metainfo interface - 'hasFetchMyTrades': true, - 'hasFetchOrder': true, - 'hasFetchOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchTickers': true, - 'hasFetchCurrencies': true, - 'hasWithdraw': true, - 'hasFetchOHLCV': true, // new metainfo interface 'has': { 'fetchOHLCV': true, From e7f6682773a9d377bc090b1f13a2da11cad4ac96 Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 06:44:43 +0300 Subject: [PATCH 30/38] removed obsolete metainfo interface for ALL the rest of the exchanges (replaced it with new metainfo interface)... --- js/_1broker.js | 10 ++++++---- js/_1btcxe.js | 8 +++++--- js/acx.js | 10 ++++++---- js/allcoin.js | 4 +++- js/anxpro.js | 8 +++++--- js/bit2c.js | 4 +++- js/bitbay.js | 6 ++++-- js/bitcoincoid.js | 3 +-- js/bitfinex2.js | 11 +---------- js/bitflyer.js | 6 ++++-- js/bithumb.js | 6 +----- js/bitlish.js | 10 ++++++---- js/bitmarket.js | 8 +++++--- js/bitmex.js | 8 +++++--- js/bitso.js | 4 +++- js/bitstamp.js | 6 +----- js/bitstamp1.js | 4 +++- js/bittrex.js | 12 +----------- js/bl3p.js | 4 +++- js/bleutrade.js | 8 +++++--- js/braziliex.js | 5 ----- js/btcbox.js | 6 ++++-- js/btcchina.js | 4 +++- js/btcexchange.js | 4 +++- js/btcmarkets.js | 7 +------ js/btctradeua.js | 4 +++- js/btcturk.js | 8 +++++--- js/btcx.js | 4 +++- js/bter.js | 5 ----- js/bxinth.js | 6 ++++-- js/ccex.js | 6 ++++-- js/cex.js | 10 ++++++---- js/chbtc.js | 6 ++++-- js/chilebit.js | 4 +++- js/coincheck.js | 4 +++- js/coinfloor.js | 4 +++- js/coingi.js | 6 ++++-- js/coinmate.js | 4 +++- js/coinspot.js | 4 +++- js/cryptopia.js | 13 +------------ js/dsx.js | 16 +++++++++------- js/exmo.js | 8 +++++--- js/flowbtc.js | 4 +++- js/foxbit.js | 4 +++- js/fybse.js | 4 +++- js/fybsg.js | 4 +++- js/gatecoin.js | 8 +++++--- js/gdax.js | 10 ---------- js/gemini.js | 4 ---- js/hitbtc.js | 13 +++++++------ js/hitbtc2.js | 13 +------------ js/huobi.js | 6 ++++-- js/huobicny.js | 4 +++- js/huobipro.js | 7 +------ js/independentreserve.js | 4 +++- js/itbit.js | 4 +++- js/jubi.js | 6 ++++-- js/kraken.js | 12 +----------- js/kucoin.js | 13 +------------ js/kuna.js | 8 +++++--- js/lakebtc.js | 4 +++- js/liqui.js | 11 +---------- js/livecoin.js | 6 +----- js/luno.js | 4 +--- js/lykke.js | 6 +----- js/mercado.js | 6 ++++-- js/mixcoins.js | 4 +++- js/nova.js | 4 +++- js/okcoincny.js | 4 +++- js/okcoinusd.js | 10 +--------- js/okex.js | 6 ++++-- js/paymium.js | 4 +++- js/poloniex.js | 3 +-- js/qryptos.js | 4 ++-- js/quadrigacx.js | 5 +---- js/quoine.js | 6 ++++-- js/southxchange.js | 8 +++++--- js/surbitcoin.js | 4 +++- js/test/test.js | 18 +++++++++--------- js/therock.js | 6 ++++-- js/tidex.js | 6 ++++-- js/urdubit.js | 4 +++- js/vaultoro.js | 4 +++- js/vbtc.js | 4 +++- js/virwox.js | 4 +++- js/wex.js | 6 ++++-- js/xbtce.js | 10 ++++++---- js/yobit.js | 6 ++++-- js/yunbi.js | 8 +++++--- js/zaif.js | 10 ++++++---- js/zb.js | 6 ++++-- 91 files changed, 297 insertions(+), 300 deletions(-) diff --git a/js/_1broker.js b/js/_1broker.js index 14dbc02a3e3e1..4fc76b5db2ac1 100644 --- a/js/_1broker.js +++ b/js/_1broker.js @@ -16,10 +16,12 @@ module.exports = class _1broker extends Exchange { 'countries': 'US', 'rateLimit': 1500, 'version': 'v2', - 'hasPublicAPI': false, - 'hasCORS': true, - 'hasFetchTrades': false, - 'hasFetchOHLCV': true, + 'has': { + 'publicAPI': false, + 'CORS': true, + 'fetchTrades': false, + 'fetchOHLCV': true, + }, 'timeframes': { '1m': '60', '15m': '900', diff --git a/js/_1btcxe.js b/js/_1btcxe.js index 9984ecc7497d9..87f2492f4b801 100644 --- a/js/_1btcxe.js +++ b/js/_1btcxe.js @@ -15,9 +15,11 @@ module.exports = class _1btcxe extends Exchange { 'name': '1BTCXE', 'countries': 'PA', // Panama 'comment': 'Crypto Capital API', - 'hasCORS': true, - 'hasFetchOHLCV': true, - 'hasWithdraw': true, + 'has': { + 'CORS': true, + 'fetchTickers': true, + 'withdraw': true, + }, 'timeframes': { '1d': '1year', }, diff --git a/js/acx.js b/js/acx.js index c0ca3ebbe7e87..cfd0727f52d4d 100644 --- a/js/acx.js +++ b/js/acx.js @@ -16,10 +16,12 @@ module.exports = class acx extends Exchange { 'countries': 'AU', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': true, - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, - 'hasWithdraw': true, + 'has': { + 'CORS': true, + 'fetchTickers': true, + 'fetchOHLCV': true, + 'withdraw': true, + }, 'timeframes': { '1m': '1', '5m': '5', diff --git a/js/allcoin.js b/js/allcoin.js index 934dc133c659f..df406f659b9da 100644 --- a/js/allcoin.js +++ b/js/allcoin.js @@ -13,7 +13,9 @@ module.exports = class allcoin extends okcoinusd { 'id': 'allcoin', 'name': 'Allcoin', 'countries': 'CA', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'extension': '', 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/31561809-c316b37c-b061-11e7-8d5a-b547b4d730eb.jpg', diff --git a/js/anxpro.js b/js/anxpro.js index 705a1ebd2c905..901dae792aa06 100644 --- a/js/anxpro.js +++ b/js/anxpro.js @@ -16,9 +16,11 @@ module.exports = class anxpro extends Exchange { 'countries': [ 'JP', 'SG', 'HK', 'NZ' ], 'version': '2', 'rateLimit': 1500, - 'hasCORS': false, - 'hasFetchTrades': false, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'fetchTrades': false, + 'withdraw': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27765983-fd8595da-5ec9-11e7-82e3-adb3ab8c2612.jpg', 'api': 'https://anxpro.com/api', diff --git a/js/bit2c.js b/js/bit2c.js index ee30d7ece5764..8c1bcbf8181d2 100644 --- a/js/bit2c.js +++ b/js/bit2c.js @@ -14,7 +14,9 @@ module.exports = class bit2c extends Exchange { 'name': 'Bit2C', 'countries': 'IL', // Israel 'rateLimit': 3000, - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766119-3593220e-5ece-11e7-8b3a-5a041f6bcc3f.jpg', 'api': 'https://www.bit2c.co.il', diff --git a/js/bitbay.js b/js/bitbay.js index 020fa9b6275fa..199ecc8f3c1d6 100644 --- a/js/bitbay.js +++ b/js/bitbay.js @@ -15,8 +15,10 @@ module.exports = class bitbay extends Exchange { 'name': 'BitBay', 'countries': [ 'PL', 'EU' ], // Poland 'rateLimit': 1000, - 'hasCORS': true, - 'hasWithdraw': true, + 'has': { + 'CORS': true, + 'withdraw': true + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766132-978a7bd8-5ece-11e7-9540-bc96d1e9bbb8.jpg', 'www': 'https://bitbay.net', diff --git a/js/bitcoincoid.js b/js/bitcoincoid.js index 051df4a8eb41a..6463b34ebbc71 100644 --- a/js/bitcoincoid.js +++ b/js/bitcoincoid.js @@ -14,9 +14,8 @@ module.exports = class bitcoincoid extends Exchange { 'id': 'bitcoincoid', 'name': 'Bitcoin.co.id', 'countries': 'ID', // Indonesia - 'hasCORS': false, - // new metainfo interface 'has': { + 'CORS': false, 'fetchTickers': false, 'fetchOHLCV': false, 'fetchOrder': true, diff --git a/js/bitfinex2.js b/js/bitfinex2.js index 001acee35ad0e..7848d0be1673e 100644 --- a/js/bitfinex2.js +++ b/js/bitfinex2.js @@ -15,18 +15,9 @@ module.exports = class bitfinex2 extends bitfinex { 'name': 'Bitfinex v2', 'countries': 'VG', 'version': 'v2', - 'hasCORS': true, - // old metainfo interface - 'hasCreateOrder': false, - 'hasFetchOrder': true, - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, - 'hasWithdraw': true, - 'hasDeposit': false, - 'hasFetchOpenOrders': false, - 'hasFetchClosedOrders': false, // new metainfo interface 'has': { + 'CORS': true, 'createOrder': false, 'fetchOHLCV': true, 'fetchTickers': true, diff --git a/js/bitflyer.js b/js/bitflyer.js index 4d4fc83f94d33..7782aac80e1a0 100644 --- a/js/bitflyer.js +++ b/js/bitflyer.js @@ -15,8 +15,10 @@ module.exports = class bitflyer extends Exchange { 'countries': 'JP', 'version': 'v1', 'rateLimit': 500, - 'hasCORS': false, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'withdraw': true + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28051642-56154182-660e-11e7-9b0d-6042d1e6edd8.jpg', 'api': 'https://api.bitflyer.jp', diff --git a/js/bithumb.js b/js/bithumb.js index 50be23224f15c..d08287f6ccbed 100644 --- a/js/bithumb.js +++ b/js/bithumb.js @@ -15,12 +15,8 @@ module.exports = class bithumb extends Exchange { 'name': 'Bithumb', 'countries': 'KR', // South Korea 'rateLimit': 500, - 'hasCORS': true, - // obsolete metainfo interface - 'hasFetchTickers': true, - 'hasWithdraw': true, - // new metainfo interface 'has': { + 'CORS': true, 'fetchTickers': true, 'withdraw': true, }, diff --git a/js/bitlish.js b/js/bitlish.js index f2ec45cc74679..2e077a7d1462e 100644 --- a/js/bitlish.js +++ b/js/bitlish.js @@ -16,10 +16,12 @@ module.exports = class bitlish extends Exchange { 'countries': [ 'GB', 'EU', 'RU' ], 'rateLimit': 1500, 'version': 'v1', - 'hasCORS': false, - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'fetchTickers': true, + 'fetchOHLCV': true, + 'withdraw': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766275-dcfc6c30-5ed3-11e7-839d-00a846385d0b.jpg', 'api': 'https://bitlish.com/api', diff --git a/js/bitmarket.js b/js/bitmarket.js index ef7a2c8f46b1b..d456079500e45 100644 --- a/js/bitmarket.js +++ b/js/bitmarket.js @@ -15,9 +15,11 @@ module.exports = class bitmarket extends Exchange { 'name': 'BitMarket', 'countries': [ 'PL', 'EU' ], 'rateLimit': 1500, - 'hasCORS': false, - 'hasFetchOHLCV': true, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'fetchOHLCV': true, + 'withdraw': true, + }, 'timeframes': { '90m': '90m', '6h': '6h', diff --git a/js/bitmex.js b/js/bitmex.js index d6b6bf364578a..2c3cda0de53ac 100644 --- a/js/bitmex.js +++ b/js/bitmex.js @@ -17,9 +17,11 @@ module.exports = class bitmex extends Exchange { 'version': 'v1', 'userAgent': undefined, 'rateLimit': 1500, - 'hasCORS': false, - 'hasFetchOHLCV': true, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'fetchOHLCV': true, + 'withdraw': true, + }, 'timeframes': { '1m': '1m', '5m': '5m', diff --git a/js/bitso.js b/js/bitso.js index f9f48316352ca..472574a82c30f 100644 --- a/js/bitso.js +++ b/js/bitso.js @@ -16,7 +16,9 @@ module.exports = class bitso extends Exchange { 'countries': 'MX', // Mexico 'rateLimit': 2000, // 30 requests per minute 'version': 'v3', - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766335-715ce7aa-5ed5-11e7-88a8-173a27bb30fe.jpg', 'api': 'https://api.bitso.com', diff --git a/js/bitstamp.js b/js/bitstamp.js index 8db0e20aed7e0..db2f2536af9a1 100644 --- a/js/bitstamp.js +++ b/js/bitstamp.js @@ -16,12 +16,8 @@ module.exports = class bitstamp extends Exchange { 'countries': 'GB', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': false, - // obsolete metainfo interface - 'hasFetchOrder': true, - 'hasWithdraw': true, - // new metainfo interface 'has': { + 'CORS': true, 'fetchOrder': true, 'withdraw': true, }, diff --git a/js/bitstamp1.js b/js/bitstamp1.js index 46c69dc1d48b7..a9474bfb6bfac 100644 --- a/js/bitstamp1.js +++ b/js/bitstamp1.js @@ -16,7 +16,9 @@ module.exports = class bitstamp1 extends Exchange { 'countries': 'GB', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27786377-8c8ab57e-5fe9-11e7-8ea4-2b05b6bcceec.jpg', 'api': 'https://www.bitstamp.net/api', diff --git a/js/bittrex.js b/js/bittrex.js index b7cfbbe252043..ca27dd2982ac4 100644 --- a/js/bittrex.js +++ b/js/bittrex.js @@ -17,19 +17,9 @@ module.exports = class bittrex extends Exchange { 'version': 'v1.1', 'rateLimit': 1500, 'hasAlreadyAuthenticatedSuccessfully': false, // a workaround for APIKEY_INVALID - 'hasCORS': false, - // obsolete metainfo interface - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, - 'hasFetchOrder': true, - 'hasFetchOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchMyTrades': false, - 'hasFetchCurrencies': true, - 'hasWithdraw': true, // new metainfo interface 'has': { + 'CORS': true, 'fetchTickers': true, 'fetchOHLCV': true, 'fetchOrder': true, diff --git a/js/bl3p.js b/js/bl3p.js index bc7b4bf01f566..85d8ff02ece4c 100644 --- a/js/bl3p.js +++ b/js/bl3p.js @@ -16,7 +16,9 @@ module.exports = class bl3p extends Exchange { 'rateLimit': 1000, 'version': '1', 'comment': 'An exchange market by BitonicNL', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28501752-60c21b82-6feb-11e7-818b-055ee6d0e754.jpg', 'api': 'https://api.bl3p.eu', diff --git a/js/bleutrade.js b/js/bleutrade.js index 28909fd1bf656..c0625f20630d2 100644 --- a/js/bleutrade.js +++ b/js/bleutrade.js @@ -15,9 +15,11 @@ module.exports = class bleutrade extends bittrex { 'countries': 'BR', // Brazil 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': true, - 'hasFetchTickers': true, - 'hasFetchOHLCV': false, + 'has': { + 'CORS': true, + 'fetchTickers': true, + 'fetchOHLCV': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30303000-b602dbe6-976d-11e7-956d-36c5049c01e7.jpg', 'api': { diff --git a/js/braziliex.js b/js/braziliex.js index 7d52681d596bf..1ce5a4ec354dd 100644 --- a/js/braziliex.js +++ b/js/braziliex.js @@ -15,11 +15,6 @@ module.exports = class braziliex extends Exchange { 'name': 'Braziliex', 'countries': 'BR', 'rateLimit': 1000, - // obsolete metainfo interface - 'hasFetchTickers': true, - 'hasFetchOpenOrders': true, - 'hasFetchMyTrades': true, - // new metainfo interface 'has': { 'fetchTickers': true, 'fetchOpenOrders': true, diff --git a/js/btcbox.js b/js/btcbox.js index 5c8438dfae9f3..964ae6def6fd1 100644 --- a/js/btcbox.js +++ b/js/btcbox.js @@ -16,8 +16,10 @@ module.exports = class btcbox extends Exchange { 'countries': 'JP', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': false, - 'hasFetchOHLCV': false, + 'has': { + 'CORS': false, + 'fetchOHLCV': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/31275803-4df755a8-aaa1-11e7-9abb-11ec2fad9f2d.jpg', 'api': 'https://www.btcbox.co.jp/api', diff --git a/js/btcchina.js b/js/btcchina.js index 78acf6549620f..5f24ecf5ed256 100644 --- a/js/btcchina.js +++ b/js/btcchina.js @@ -15,7 +15,9 @@ module.exports = class btcchina extends Exchange { 'countries': 'CN', 'rateLimit': 1500, 'version': 'v1', - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766368-465b3286-5ed6-11e7-9a11-0f6467e1d82b.jpg', 'api': { diff --git a/js/btcexchange.js b/js/btcexchange.js index deda1ce00b531..d0e8e3f22d612 100644 --- a/js/btcexchange.js +++ b/js/btcexchange.js @@ -14,7 +14,9 @@ module.exports = class btcexchange extends btcturk { 'name': 'BTCExchange', 'countries': 'PH', // Philippines 'rateLimit': 1500, - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27993052-4c92911a-64aa-11e7-96d8-ec6ac3435757.jpg', 'api': 'https://www.btcexchange.ph/api', diff --git a/js/btcmarkets.js b/js/btcmarkets.js index 4534a36b6d0c9..e33e964effd0f 100644 --- a/js/btcmarkets.js +++ b/js/btcmarkets.js @@ -15,13 +15,8 @@ module.exports = class btcmarkets extends Exchange { 'name': 'BTC Markets', 'countries': 'AU', // Australia 'rateLimit': 1000, // market data cached for 1 second (trades cached for 2 seconds) - 'hasCORS': false, - 'hasFetchOrder': true, - 'hasFetchOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchMyTrades': true, 'has': { + 'CORS': false, 'fetchOrder': true, 'fetchOrders': true, 'fetchClosedOrders': 'emulated', diff --git a/js/btctradeua.js b/js/btctradeua.js index df03f81104790..741882e4b1c34 100644 --- a/js/btctradeua.js +++ b/js/btctradeua.js @@ -15,7 +15,9 @@ module.exports = class btctradeua extends Exchange { 'name': 'BTC Trade UA', 'countries': 'UA', // Ukraine, 'rateLimit': 3000, - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27941483-79fc7350-62d9-11e7-9f61-ac47f28fcd96.jpg', 'api': 'https://btc-trade.com.ua/api', diff --git a/js/btcturk.js b/js/btcturk.js index 229a62530c7b3..04d889f98bf83 100644 --- a/js/btcturk.js +++ b/js/btcturk.js @@ -15,9 +15,11 @@ module.exports = class btcturk extends Exchange { 'name': 'BTCTurk', 'countries': 'TR', // Turkey 'rateLimit': 1000, - 'hasCORS': true, - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, + 'has': { + 'CORS': true, + 'fetchTickers': true, + 'fetchOHLCV': true, + }, 'timeframes': { '1d': '1d', }, diff --git a/js/btcx.js b/js/btcx.js index 7e80ecd870ed1..a666868f2f20c 100644 --- a/js/btcx.js +++ b/js/btcx.js @@ -16,7 +16,9 @@ module.exports = class btcx extends Exchange { 'countries': [ 'IS', 'US', 'EU' ], 'rateLimit': 1500, // support in english is very poor, unable to tell rate limits 'version': 'v1', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766385-9fdcc98c-5ed6-11e7-8f14-66d5e5cd47e6.jpg', 'api': 'https://btc-x.is/api', diff --git a/js/bter.js b/js/bter.js index 540aa2841d148..dc32b23340664 100644 --- a/js/bter.js +++ b/js/bter.js @@ -15,11 +15,6 @@ module.exports = class bter extends Exchange { 'name': 'Bter', 'countries': [ 'VG', 'CN' ], // British Virgin Islands, China 'version': '2', - // obsolete metainfo interface - 'hasCORS': false, - 'hasFetchTickers': true, - 'hasWithdraw': true, - // new metainfo interface 'has': { 'CORS': false, 'fetchTickers': true, diff --git a/js/bxinth.js b/js/bxinth.js index 15ec71cb4568d..bb081e9109ce9 100644 --- a/js/bxinth.js +++ b/js/bxinth.js @@ -15,8 +15,10 @@ module.exports = class bxinth extends Exchange { 'name': 'BX.in.th', 'countries': 'TH', // Thailand 'rateLimit': 1500, - 'hasCORS': false, - 'hasFetchTickers': true, + 'has': { + 'CORS': false, + 'fetchTickers': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766412-567b1eb4-5ed7-11e7-94a8-ff6a3884f6c5.jpg', 'api': 'https://bx.in.th/api', diff --git a/js/ccex.js b/js/ccex.js index 462662b1d2ea3..d738eede293a3 100644 --- a/js/ccex.js +++ b/js/ccex.js @@ -15,8 +15,10 @@ module.exports = class ccex extends Exchange { 'name': 'C-CEX', 'countries': [ 'DE', 'EU' ], 'rateLimit': 1500, - 'hasCORS': false, - 'hasFetchTickers': true, + 'has': { + 'CORS': false, + 'fetchTickers': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766433-16881f90-5ed8-11e7-92f8-3d92cc747a6c.jpg', 'api': { diff --git a/js/cex.js b/js/cex.js index 42aeba2025158..6bcc8ee19875f 100644 --- a/js/cex.js +++ b/js/cex.js @@ -15,10 +15,12 @@ module.exports = class cex extends Exchange { 'name': 'CEX.IO', 'countries': [ 'GB', 'EU', 'CY', 'RU' ], 'rateLimit': 1500, - 'hasCORS': true, - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, - 'hasFetchOpenOrders': true, + 'has': { + 'CORS': true, + 'fetchTickers': true, + 'fetchOHLCV': true, + 'fetchOpenOrders': true, + }, 'timeframes': { '1m': '1m', }, diff --git a/js/chbtc.js b/js/chbtc.js index fbf22c6f4b5ab..094db9c34b92a 100644 --- a/js/chbtc.js +++ b/js/chbtc.js @@ -16,8 +16,10 @@ module.exports = class chbtc extends zb { 'countries': 'CN', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': false, - 'hasFetchOrder': true, + 'has': { + 'CORS': false, + 'fetchOrder': true + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28555659-f0040dc2-7109-11e7-9d99-688a438bf9f4.jpg', 'api': { diff --git a/js/chilebit.js b/js/chilebit.js index 979ede9a40690..5faa9ac5a9d30 100644 --- a/js/chilebit.js +++ b/js/chilebit.js @@ -13,7 +13,9 @@ module.exports = class chilebit extends foxbit { 'id': 'chilebit', 'name': 'ChileBit', 'countries': 'CL', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27991414-1298f0d8-647f-11e7-9c40-d56409266336.jpg', 'api': { diff --git a/js/coincheck.js b/js/coincheck.js index 41443c1b69c78..2cd8630918455 100644 --- a/js/coincheck.js +++ b/js/coincheck.js @@ -15,7 +15,9 @@ module.exports = class coincheck extends Exchange { 'name': 'coincheck', 'countries': [ 'JP', 'ID' ], 'rateLimit': 1500, - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766464-3b5c3c74-5ed9-11e7-840e-31b32968e1da.jpg', 'api': 'https://coincheck.com/api', diff --git a/js/coinfloor.js b/js/coinfloor.js index 48b88a0f33b03..c9949578b1926 100644 --- a/js/coinfloor.js +++ b/js/coinfloor.js @@ -15,7 +15,9 @@ module.exports = class coinfloor extends Exchange { 'name': 'coinfloor', 'rateLimit': 1000, 'countries': 'UK', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28246081-623fc164-6a1c-11e7-913f-bac0d5576c90.jpg', 'api': 'https://webapi.coinfloor.co.uk:8090/bist', diff --git a/js/coingi.js b/js/coingi.js index 0725d1121075f..f039f3d947a06 100644 --- a/js/coingi.js +++ b/js/coingi.js @@ -15,8 +15,10 @@ module.exports = class coingi extends Exchange { 'name': 'Coingi', 'rateLimit': 1000, 'countries': [ 'PA', 'BG', 'CN', 'US' ], // Panama, Bulgaria, China, US - 'hasFetchTickers': true, - 'hasCORS': false, + 'has': { + 'CORS': false, + 'fetchTickers': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28619707-5c9232a8-7212-11e7-86d6-98fe5d15cc6e.jpg', 'api': { diff --git a/js/coinmate.js b/js/coinmate.js index d6b98cc999d61..be28841989585 100644 --- a/js/coinmate.js +++ b/js/coinmate.js @@ -15,7 +15,9 @@ module.exports = class coinmate extends Exchange { 'name': 'CoinMate', 'countries': [ 'GB', 'CZ', 'EU' ], // UK, Czech Republic 'rateLimit': 1000, - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27811229-c1efb510-606c-11e7-9a36-84ba2ce412d8.jpg', 'api': 'https://coinmate.io/api', diff --git a/js/coinspot.js b/js/coinspot.js index 7a9866b48130f..859019f8bf432 100644 --- a/js/coinspot.js +++ b/js/coinspot.js @@ -15,7 +15,9 @@ module.exports = class coinspot extends Exchange { 'name': 'CoinSpot', 'countries': 'AU', // Australia 'rateLimit': 1000, - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28208429-3cacdf9a-6896-11e7-854e-4c79a772a30f.jpg', 'api': { diff --git a/js/cryptopia.js b/js/cryptopia.js index 24f0efba16959..069072a4b551c 100644 --- a/js/cryptopia.js +++ b/js/cryptopia.js @@ -15,19 +15,8 @@ module.exports = class cryptopia extends Exchange { 'name': 'Cryptopia', 'rateLimit': 1500, 'countries': 'NZ', // New Zealand - 'hasCORS': false, - // obsolete metainfo interface - 'hasFetchTickers': true, - 'hasFetchOrder': true, - 'hasFetchOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchMyTrades': true, - 'hasFetchCurrencies': true, - 'hasDeposit': true, - 'hasWithdraw': true, - // new metainfo interface 'has': { + 'CORS': false, 'fetchTickers': true, 'fetchOrder': 'emulated', 'fetchOrders': 'emulated', diff --git a/js/dsx.js b/js/dsx.js index 04ff347066cf5..72d824cf11937 100644 --- a/js/dsx.js +++ b/js/dsx.js @@ -14,13 +14,15 @@ module.exports = class dsx extends liqui { 'name': 'DSX', 'countries': 'UK', 'rateLimit': 1500, - 'hasCORS': false, - 'hasFetchOrder': true, - 'hasFetchOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchTickers': true, - 'hasFetchMyTrades': true, + 'has': { + 'CORS': false, + 'fetchOrder': true, + 'fetchOrders': true, + 'fetchOpenOrders': true, + 'fetchClosedOrders': true, + 'fetchTickers': true, + 'fetchMyTrades': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27990275-1413158a-645a-11e7-931c-94717f7510e3.jpg', 'api': { diff --git a/js/exmo.js b/js/exmo.js index d56b014e6b06d..f07c64bfa203a 100644 --- a/js/exmo.js +++ b/js/exmo.js @@ -16,9 +16,11 @@ module.exports = class exmo extends Exchange { 'countries': [ 'ES', 'RU' ], // Spain, Russia 'rateLimit': 1000, // once every 350 ms ≈ 180 requests per minute ≈ 3 requests per second 'version': 'v1', - 'hasCORS': false, - 'hasFetchTickers': true, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'fetchTickers': true, + 'withdraw': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766491-1b0ea956-5eda-11e7-9225-40d67b481b8d.jpg', 'api': 'https://api.exmo.com', diff --git a/js/flowbtc.js b/js/flowbtc.js index 1ea14e90c79bc..a4845c426b639 100644 --- a/js/flowbtc.js +++ b/js/flowbtc.js @@ -16,7 +16,9 @@ module.exports = class flowbtc extends Exchange { 'countries': 'BR', // Brazil 'version': 'v1', 'rateLimit': 1000, - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28162465-cd815d4c-67cf-11e7-8e57-438bea0523a2.jpg', 'api': 'https://api.flowbtc.com:8400/ajax', diff --git a/js/foxbit.js b/js/foxbit.js index ab904a6a601c4..2805c05308be1 100644 --- a/js/foxbit.js +++ b/js/foxbit.js @@ -14,7 +14,9 @@ module.exports = class foxbit extends Exchange { 'id': 'foxbit', 'name': 'FoxBit', 'countries': 'BR', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'rateLimit': 1000, 'version': 'v1', 'urls': { diff --git a/js/fybse.js b/js/fybse.js index 00bd09ca3bb57..33670f0801e78 100644 --- a/js/fybse.js +++ b/js/fybse.js @@ -14,7 +14,9 @@ module.exports = class fybse extends Exchange { 'id': 'fybse', 'name': 'FYB-SE', 'countries': 'SE', // Sweden - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'rateLimit': 1500, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766512-31019772-5edb-11e7-8241-2e675e6797f1.jpg', diff --git a/js/fybsg.js b/js/fybsg.js index 819309eead992..a675dd2eaee6b 100644 --- a/js/fybsg.js +++ b/js/fybsg.js @@ -13,7 +13,9 @@ module.exports = class fybsg extends fybse { 'id': 'fybsg', 'name': 'FYB-SG', 'countries': 'SG', // Singapore - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766513-3364d56a-5edb-11e7-9e6b-d5898bb89c81.jpg', 'api': 'https://www.fybsg.com/api/SGD', diff --git a/js/gatecoin.js b/js/gatecoin.js index 12d5559e7ec7a..303c053316e93 100644 --- a/js/gatecoin.js +++ b/js/gatecoin.js @@ -16,9 +16,11 @@ module.exports = class gatecoin extends Exchange { 'rateLimit': 2000, 'countries': 'HK', // Hong Kong 'comment': 'a regulated/licensed exchange', - 'hasCORS': false, - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, + 'has': { + 'CORS': false, + 'fetchTickers': true, + 'fetchOHLCV': true, + }, 'timeframes': { '1m': '1m', '15m': '15m', diff --git a/js/gdax.js b/js/gdax.js index fb0b39790edc4..1714efab02d5c 100644 --- a/js/gdax.js +++ b/js/gdax.js @@ -16,16 +16,6 @@ module.exports = class gdax extends Exchange { 'countries': 'US', 'rateLimit': 1000, 'userAgent': this.userAgents['chrome'], - // obsolete metainfo interface - 'hasCORS': true, - 'hasFetchOHLCV': true, - 'hasDeposit': true, - 'hasWithdraw': true, - 'hasFetchOrder': true, - 'hasFetchOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - // new metainfo interface 'has': { 'CORS': true, 'fetchOHLCV': true, diff --git a/js/gemini.js b/js/gemini.js index cae6dac65d20b..0367cbaa4be66 100644 --- a/js/gemini.js +++ b/js/gemini.js @@ -16,10 +16,6 @@ module.exports = class gemini extends Exchange { 'countries': 'US', 'rateLimit': 1500, // 200 for private API 'version': 'v1', - // obsolete metainfo interface - 'hasCORS': false, - 'hasWithdraw': true, - // new metainfo interface 'has': { 'CORS': false, 'withdraw': true, diff --git a/js/hitbtc.js b/js/hitbtc.js index d681f6ed291d5..2840db29949b8 100644 --- a/js/hitbtc.js +++ b/js/hitbtc.js @@ -16,12 +16,13 @@ module.exports = class hitbtc extends Exchange { 'countries': 'UK', 'rateLimit': 1500, 'version': '1', - 'hasCORS': false, - 'hasFetchTickers': true, - 'hasFetchOrder': true, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'fetchOrder': true, + 'fetchOpenOrders': true, + 'fetchClosedOrders': true, + 'withdraw': true + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766555-8eaec20e-5edc-11e7-9c5b-6dc69fc42f5e.jpg', 'api': 'http://api.hitbtc.com', diff --git a/js/hitbtc2.js b/js/hitbtc2.js index 7d7550c531e2d..1081d70ced86b 100644 --- a/js/hitbtc2.js +++ b/js/hitbtc2.js @@ -16,19 +16,8 @@ module.exports = class hitbtc2 extends hitbtc { 'countries': 'UK', 'rateLimit': 1500, 'version': '2', - 'hasCORS': true, - // older metainfo interface - 'hasFetchOHLCV': true, - 'hasFetchTickers': true, - 'hasFetchOrder': true, - 'hasFetchOrders': false, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchMyTrades': true, - 'hasWithdraw': true, - 'hasFetchCurrencies': true, - // new metainfo interface 'has': { + 'CORS': true, 'fetchCurrencies': true, 'fetchOHLCV': true, 'fetchTickers': true, diff --git a/js/huobi.js b/js/huobi.js index 3f3d78ff66a96..9cf998ee0de0b 100644 --- a/js/huobi.js +++ b/js/huobi.js @@ -16,8 +16,10 @@ module.exports = class huobi extends Exchange { 'countries': 'CN', 'rateLimit': 2000, 'version': 'v3', - 'hasCORS': false, - 'hasFetchOHLCV': true, + 'has': { + 'CORS': false, + 'fetchOHLCV': true, + }, 'timeframes': { '1m': '001', '5m': '005', diff --git a/js/huobicny.js b/js/huobicny.js index 690c9028fea70..631a67c719f3f 100644 --- a/js/huobicny.js +++ b/js/huobicny.js @@ -13,7 +13,9 @@ module.exports = class huobicny extends huobipro { 'id': 'huobicny', 'name': 'Huobi CNY', 'hostname': 'be.huobi.com', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766569-15aa7b9a-5edd-11e7-9e7f-44791f4ee49c.jpg', 'api': 'https://be.huobi.com', diff --git a/js/huobipro.js b/js/huobipro.js index b271f40450078..f787b885a8747 100644 --- a/js/huobipro.js +++ b/js/huobipro.js @@ -20,13 +20,8 @@ module.exports = class huobipro extends Exchange { 'accounts': undefined, 'accountsById': undefined, 'hostname': 'api.huobi.pro', - 'hasCORS': false, - // obsolete metainfo structure - 'hasFetchOHLCV': true, - 'hasFetchOrders': true, - 'hasFetchOpenOrders': true, - // new metainfo structure 'has': { + 'CORS': false, 'fetchOHCLV': true, 'fetchOrders': true, 'fetchOpenOrders': true, diff --git a/js/independentreserve.js b/js/independentreserve.js index 15d4688dd26d2..f4d6c6c47527f 100644 --- a/js/independentreserve.js +++ b/js/independentreserve.js @@ -14,7 +14,9 @@ module.exports = class independentreserve extends Exchange { 'name': 'Independent Reserve', 'countries': [ 'AU', 'NZ' ], // Australia, New Zealand 'rateLimit': 1000, - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30521662-cf3f477c-9bcb-11e7-89bc-d1ac85012eda.jpg', 'api': { diff --git a/js/itbit.js b/js/itbit.js index e5477f4fe0835..0f4a81ecaa3a2 100644 --- a/js/itbit.js +++ b/js/itbit.js @@ -16,7 +16,9 @@ module.exports = class itbit extends Exchange { 'countries': 'US', 'rateLimit': 2000, 'version': 'v1', - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27822159-66153620-60ad-11e7-89e7-005f6d7f3de0.jpg', 'api': 'https://api.itbit.com', diff --git a/js/jubi.js b/js/jubi.js index b3170e53f3401..990f1a6072bee 100644 --- a/js/jubi.js +++ b/js/jubi.js @@ -15,8 +15,10 @@ module.exports = class jubi extends btcbox { 'countries': 'CN', 'rateLimit': 1500, 'version': 'v1', - 'hasCORS': false, - 'hasFetchTickers': true, + 'has': { + 'CORS': false, + 'fetchTickers': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766581-9d397d9a-5edd-11e7-8fb9-5d8236c0e692.jpg', 'api': 'https://www.jubi.com/api', diff --git a/js/kraken.js b/js/kraken.js index a52a22ebc14a1..697be05573347 100644 --- a/js/kraken.js +++ b/js/kraken.js @@ -16,18 +16,8 @@ module.exports = class kraken extends Exchange { 'countries': 'US', 'version': '0', 'rateLimit': 3000, - 'hasCORS': false, - // obsolete metainfo interface - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, - 'hasFetchOrder': true, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchMyTrades': true, - 'hasWithdraw': true, - 'hasFetchCurrencies': true, - // new metainfo interface 'has': { + 'CORS': false, 'fetchCurrencies': true, 'fetchTickers': true, 'fetchOHLCV': true, diff --git a/js/kucoin.js b/js/kucoin.js index c55b373f1065e..1f3e8af051064 100644 --- a/js/kucoin.js +++ b/js/kucoin.js @@ -16,20 +16,9 @@ module.exports = class kucoin extends Exchange { 'countries': 'HK', // Hong Kong 'version': 'v1', 'rateLimit': 2000, - 'hasCORS': false, 'userAgent': this.userAgents['chrome'], - // obsolete metainfo interface - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, - 'hasFetchOrder': false, - 'hasFetchOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchMyTrades': false, - 'hasFetchCurrencies': true, - 'hasWithdraw': true, - // new metainfo interface 'has': { + 'CORS': false, 'fetchTickers': true, 'fetchOHLCV': true, // see the method implementation below 'fetchOrder': false, diff --git a/js/kuna.js b/js/kuna.js index 2593a07f350f8..0ce987530f136 100644 --- a/js/kuna.js +++ b/js/kuna.js @@ -16,9 +16,11 @@ module.exports = class kuna extends acx { 'countries': 'UA', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': false, - 'hasFetchTickers': false, - 'hasFetchOHLCV': false, + 'has': { + 'CORS': false, + 'fetchTickers': false, + 'fetchOHLCV': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/31697638-912824fa-b3c1-11e7-8c36-cf9606eb94ac.jpg', 'api': 'https://kuna.io', diff --git a/js/lakebtc.js b/js/lakebtc.js index 9f038083d364d..59e9fe0249311 100644 --- a/js/lakebtc.js +++ b/js/lakebtc.js @@ -15,7 +15,9 @@ module.exports = class lakebtc extends Exchange { 'name': 'LakeBTC', 'countries': 'US', 'version': 'api_v2', - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28074120-72b7c38a-6660-11e7-92d9-d9027502281d.jpg', 'api': 'https://api.lakebtc.com', diff --git a/js/liqui.js b/js/liqui.js index 550d38fea3025..2f76fad35c8b6 100644 --- a/js/liqui.js +++ b/js/liqui.js @@ -11,18 +11,9 @@ module.exports = class liqui extends Exchange { 'countries': 'UA', 'rateLimit': 3000, 'version': '3', - 'hasCORS': false, 'userAgent': this.userAgents['chrome'], - // obsolete metainfo interface - 'hasFetchOrder': true, - 'hasFetchOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchTickers': true, - 'hasFetchMyTrades': true, - 'hasWithdraw': true, - // new metainfo interface 'has': { + 'CORS': false, 'fetchOrder': true, 'fetchOrders': 'emulated', 'fetchOpenOrders': true, diff --git a/js/livecoin.js b/js/livecoin.js index d73b728371eb5..2c837d323e168 100644 --- a/js/livecoin.js +++ b/js/livecoin.js @@ -15,12 +15,8 @@ module.exports = class livecoin extends Exchange { 'name': 'LiveCoin', 'countries': [ 'US', 'UK', 'RU' ], 'rateLimit': 1000, - 'hasCORS': false, - // obsolete metainfo interface - 'hasFetchTickers': true, - 'hasFetchCurrencies': true, - // new metainfo interface 'has': { + 'CORS': false, 'fetchTickers': true, 'fetchCurrencies': true, }, diff --git a/js/luno.js b/js/luno.js index b8043a019038d..49f7f816e7538 100644 --- a/js/luno.js +++ b/js/luno.js @@ -16,10 +16,8 @@ module.exports = class luno extends Exchange { 'countries': [ 'GB', 'SG', 'ZA' ], 'rateLimit': 10000, 'version': '1', - 'hasCORS': false, - 'hasFetchTickers': true, - 'hasFetchOrder': true, 'has': { + 'CORS': false, 'fetchTickers': true, 'fetchOrder': true, }, diff --git a/js/lykke.js b/js/lykke.js index 4a2a68fc76f82..e59122a419a55 100644 --- a/js/lykke.js +++ b/js/lykke.js @@ -15,12 +15,8 @@ module.exports = class lykke extends Exchange { 'countries': 'CH', 'version': 'v1', 'rateLimit': 200, - 'hasCORS': false, - // obsolete metainfo interface - 'hasFetchTrades': false, - 'hasFetchOHLCV': false, - // new metainfo interface 'has': { + 'CORS': false, 'fetchOHLCV': false, 'fetchTrades': false, }, diff --git a/js/mercado.js b/js/mercado.js index a72ff8e20540f..03d89f227ec6c 100644 --- a/js/mercado.js +++ b/js/mercado.js @@ -16,8 +16,10 @@ module.exports = class mercado extends Exchange { 'countries': 'BR', // Brazil 'rateLimit': 1000, 'version': 'v3', - 'hasCORS': true, - 'hasWithdraw': true, + 'has': { + 'CORS': true, + 'withdraw': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27837060-e7c58714-60ea-11e7-9192-f05e86adb83f.jpg', 'api': { diff --git a/js/mixcoins.js b/js/mixcoins.js index c08c9d5900363..07d38953837b0 100644 --- a/js/mixcoins.js +++ b/js/mixcoins.js @@ -16,7 +16,9 @@ module.exports = class mixcoins extends Exchange { 'countries': [ 'GB', 'HK' ], 'rateLimit': 1500, 'version': 'v1', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30237212-ed29303c-9535-11e7-8af8-fcd381cfa20c.jpg', 'api': 'https://mixcoins.com/api', diff --git a/js/nova.js b/js/nova.js index a76a4030cdf26..be214589efe32 100644 --- a/js/nova.js +++ b/js/nova.js @@ -16,7 +16,9 @@ module.exports = class nova extends Exchange { 'countries': 'TZ', // Tanzania 'rateLimit': 2000, 'version': 'v2', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30518571-78ca0bca-9b8a-11e7-8840-64b83a4a94b2.jpg', 'api': 'https://novaexchange.com/remote', diff --git a/js/okcoincny.js b/js/okcoincny.js index fe168eec3f298..764f2005eb342 100644 --- a/js/okcoincny.js +++ b/js/okcoincny.js @@ -13,7 +13,9 @@ module.exports = class okcoincny extends okcoinusd { 'id': 'okcoincny', 'name': 'OKCoin CNY', 'countries': 'CN', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766792-8be9157a-5ee5-11e7-926c-6d69b8d3378d.jpg', 'api': { diff --git a/js/okcoinusd.js b/js/okcoinusd.js index c2e7494dbacb2..a01d2a1b42f91 100644 --- a/js/okcoinusd.js +++ b/js/okcoinusd.js @@ -14,18 +14,10 @@ module.exports = class okcoinusd extends Exchange { 'id': 'okcoinusd', 'name': 'OKCoin USD', 'countries': [ 'CN', 'US' ], - 'hasCORS': false, 'version': 'v1', 'rateLimit': 1000, // up to 3000 requests per 5 minutes ≈ 600 requests per minute ≈ 10 requests per second ≈ 100 ms - // obsolete metainfo interface - 'hasFetchOHLCV': true, - 'hasFetchOrder': true, - 'hasFetchOrders': false, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasWithdraw': true, - // new metainfo interface 'has': { + 'CORS': false, 'fetchOHLCV': true, 'fetchOrder': true, 'fetchOrders': false, diff --git a/js/okex.js b/js/okex.js index d2901affb07bc..d3e29f8a27119 100644 --- a/js/okex.js +++ b/js/okex.js @@ -13,8 +13,10 @@ module.exports = class okex extends okcoinusd { 'id': 'okex', 'name': 'OKEX', 'countries': [ 'CN', 'US' ], - 'hasCORS': false, - 'hasFutureMarkets': true, + 'has': { + 'CORS': false, + 'futureMarkets': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/32552768-0d6dd3c6-c4a6-11e7-90f8-c043b64756a7.jpg', 'api': { diff --git a/js/paymium.js b/js/paymium.js index e164af0cd45af..b011c6d13fb2f 100644 --- a/js/paymium.js +++ b/js/paymium.js @@ -16,7 +16,9 @@ module.exports = class paymium extends Exchange { 'countries': [ 'FR', 'EU' ], 'rateLimit': 2000, 'version': 'v1', - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27790564-a945a9d4-5ff9-11e7-9d2d-b635763f2f24.jpg', 'api': 'https://paymium.com/api', diff --git a/js/poloniex.js b/js/poloniex.js index fbf1abb72c698..0ee622d03577d 100644 --- a/js/poloniex.js +++ b/js/poloniex.js @@ -15,9 +15,8 @@ module.exports = class poloniex extends Exchange { 'name': 'Poloniex', 'countries': 'US', 'rateLimit': 1000, // up to 6 calls per second - 'hasCORS': true, - // new metainfo interface 'has': { + 'CORS': true, 'fetchOHLCV': true, 'fetchMyTrades': true, 'fetchOrder': 'emulated', diff --git a/js/qryptos.js b/js/qryptos.js index 73ccbbbf4f3ac..3d848b23e3de3 100644 --- a/js/qryptos.js +++ b/js/qryptos.js @@ -16,9 +16,9 @@ module.exports = class qryptos extends Exchange { 'countries': [ 'CN', 'TW' ], 'version': '2', 'rateLimit': 1000, - 'hasFetchTickers': true, - 'hasCORS': false, 'has': { + 'CORS': false, + 'fetchTickers': true, 'fetchOrder': true, 'fetchOrders': true, 'fetchOpenOrders': true, diff --git a/js/quadrigacx.js b/js/quadrigacx.js index 595ec80bdf926..3a489fc0a3802 100644 --- a/js/quadrigacx.js +++ b/js/quadrigacx.js @@ -16,11 +16,8 @@ module.exports = class quadrigacx extends Exchange { 'countries': 'CA', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': true, - // obsolete metainfo interface - 'hasWithdraw': true, - // new metainfo interface 'has': { + 'CORS': true, 'withdraw': true, }, 'urls': { diff --git a/js/quoine.js b/js/quoine.js index 2c68dee043f0b..32e512ba9773b 100644 --- a/js/quoine.js +++ b/js/quoine.js @@ -15,8 +15,10 @@ module.exports = class quoine extends qryptos { 'countries': [ 'JP', 'SG', 'VN' ], 'version': '2', 'rateLimit': 1000, - 'hasFetchTickers': true, - 'hasCORS': false, + 'has': { + 'CORS': false, + 'fetchTickers': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766844-9615a4e8-5ee8-11e7-8814-fcd004db8cdd.jpg', 'api': 'https://api.quoine.com', diff --git a/js/southxchange.js b/js/southxchange.js index 6c3b2f03e22d0..0e98dc13c8fc4 100644 --- a/js/southxchange.js +++ b/js/southxchange.js @@ -15,9 +15,11 @@ module.exports = class southxchange extends Exchange { 'name': 'SouthXchange', 'countries': 'AR', // Argentina 'rateLimit': 1000, - 'hasFetchTickers': true, - 'hasCORS': false, - 'hasWithdraw': true, + 'has': { + 'CORS': true, + 'fetchTickers': true, + 'withdraw': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27838912-4f94ec8a-60f6-11e7-9e5d-bbf9bd50a559.jpg', 'api': 'https://www.southxchange.com/api', diff --git a/js/surbitcoin.js b/js/surbitcoin.js index d6f5703844e5c..f8cb68e74dc7d 100644 --- a/js/surbitcoin.js +++ b/js/surbitcoin.js @@ -13,7 +13,9 @@ module.exports = class surbitcoin extends foxbit { 'id': 'surbitcoin', 'name': 'SurBitcoin', 'countries': 'VE', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27991511-f0a50194-6481-11e7-99b5-8f02932424cc.jpg', 'api': { diff --git a/js/test/test.js b/js/test/test.js index 44f5e1a1a7b2d..7346c85e0acb6 100644 --- a/js/test/test.js +++ b/js/test/test.js @@ -87,7 +87,7 @@ let human_value = function (price) { let testTicker = async (exchange, symbol) => { - if (exchange.hasFetchTicker) { + if (exchange.has.fetchTicker) { // log (symbol.green, 'fetching ticker...') @@ -177,7 +177,7 @@ let testOrderBook = async (exchange, symbol) => { let testTrades = async (exchange, symbol) => { - if (exchange.hasFetchTrades) { + if (exchange.has.fetchTrades) { // log (symbol.green, 'fetching trades...') @@ -196,7 +196,7 @@ let testTrades = async (exchange, symbol) => { let testTickers = async (exchange, symbol) => { - if (exchange.hasFetchTickers) { + if (exchange.has.fetchTickers) { // log ('fetching all tickers at once...') @@ -224,7 +224,7 @@ let testTickers = async (exchange, symbol) => { let testOHLCV = async (exchange, symbol) => { - if (exchange.hasFetchOHLCV) { + if (exchange.has.fetchOHLCV) { // log (symbol.green, 'fetching OHLCV...') let ohlcv = await exchange.fetchOHLCV (symbol) @@ -262,7 +262,7 @@ let testSymbol = async (exchange, symbol) => { let testOrders = async (exchange, symbol) => { - if (exchange.hasFetchOrders) { + if (exchange.has.fetchOrders) { // log ('fetching orders...') let orders = await exchange.fetchOrders (symbol) @@ -279,7 +279,7 @@ let testOrders = async (exchange, symbol) => { let testClosedOrders = async (exchange, symbol) => { - if (exchange.hasFetchClosedOrders) { + if (exchange.has.fetchClosedOrders) { // log ('fetching closed orders...') let orders = await exchange.fetchClosedOrders (symbol) @@ -296,7 +296,7 @@ let testClosedOrders = async (exchange, symbol) => { let testOpenOrders = async (exchange, symbol) => { - if (exchange.hasFetchOpenOrders) { + if (exchange.has.fetchOpenOrders) { // log ('fetching open orders...') let orders = await exchange.fetchOpenOrders (symbol) @@ -313,7 +313,7 @@ let testOpenOrders = async (exchange, symbol) => { let testMyTrades = async (exchange, symbol) => { - if (exchange.hasFetchMyTrades) { + if (exchange.has.fetchMyTrades) { // log ('fetching my trades...') let trades = await exchange.fetchMyTrades (symbol, 0) @@ -331,7 +331,7 @@ let testMyTrades = async (exchange, symbol) => { let testFetchCurrencies = async (exchange, symbol) => { - if (exchange.hasFetchCurrencies) { + if (exchange.has.fetchCurrencies) { // log ('fetching currencies...') let currencies = await exchange.fetchCurrencies () diff --git a/js/therock.js b/js/therock.js index 17388ce4677ab..9058badc850e4 100644 --- a/js/therock.js +++ b/js/therock.js @@ -16,8 +16,10 @@ module.exports = class therock extends Exchange { 'countries': 'MT', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': false, - 'hasFetchTickers': true, + 'has': { + 'CORS': false, + 'fetchTickers': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766869-75057fa2-5ee9-11e7-9a6f-13e641fa4707.jpg', 'api': 'https://api.therocktrading.com', diff --git a/js/tidex.js b/js/tidex.js index 2ebc0ddf91f34..99d2c6af1bd19 100644 --- a/js/tidex.js +++ b/js/tidex.js @@ -15,8 +15,10 @@ module.exports = class tidex extends liqui { 'countries': 'UK', 'rateLimit': 2000, 'version': '3', - // 'hasCORS': false, - // 'hasFetchTickers': true, + 'has': { + // 'CORS': false, + // 'fetchTickers': true + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30781780-03149dc4-a12e-11e7-82bb-313b269d24d4.jpg', 'api': { diff --git a/js/urdubit.js b/js/urdubit.js index c6b29f3e4109f..98e93776622cb 100644 --- a/js/urdubit.js +++ b/js/urdubit.js @@ -13,7 +13,9 @@ module.exports = class urdubit extends foxbit { 'id': 'urdubit', 'name': 'UrduBit', 'countries': 'PK', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27991453-156bf3ae-6480-11e7-82eb-7295fe1b5bb4.jpg', 'api': { diff --git a/js/vaultoro.js b/js/vaultoro.js index 119fd06bc2f1e..e8fc3f7ef6b82 100644 --- a/js/vaultoro.js +++ b/js/vaultoro.js @@ -15,7 +15,9 @@ module.exports = class vaultoro extends Exchange { 'countries': 'CH', 'rateLimit': 1000, 'version': '1', - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766880-f205e870-5ee9-11e7-8fe2-0d5b15880752.jpg', 'api': 'https://api.vaultoro.com', diff --git a/js/vbtc.js b/js/vbtc.js index 348152d32c63b..c1c17bf245a63 100644 --- a/js/vbtc.js +++ b/js/vbtc.js @@ -13,7 +13,9 @@ module.exports = class vbtc extends foxbit { 'id': 'vbtc', 'name': 'VBTC', 'countries': 'VN', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27991481-1f53d1d8-6481-11e7-884e-21d17e7939db.jpg', 'api': { diff --git a/js/virwox.js b/js/virwox.js index 38abf19cc4ab5..cf084fa1a939c 100644 --- a/js/virwox.js +++ b/js/virwox.js @@ -15,7 +15,9 @@ module.exports = class virwox extends Exchange { 'name': 'VirWoX', 'countries': [ 'AT', 'EU' ], 'rateLimit': 1000, - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766894-6da9d360-5eea-11e7-90aa-41f2711b7405.jpg', 'api': { diff --git a/js/wex.js b/js/wex.js index 675532c38716c..22478885c9091 100644 --- a/js/wex.js +++ b/js/wex.js @@ -15,8 +15,10 @@ module.exports = class wex extends liqui { 'name': 'WEX', 'countries': 'NZ', // New Zealand 'version': '3', - 'hasFetchTickers': true, - 'hasCORS': false, + 'has': { + 'CORS': false, + 'fetchTickers': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30652751-d74ec8f8-9e31-11e7-98c5-71469fcef03e.jpg', 'api': { diff --git a/js/xbtce.js b/js/xbtce.js index a7905b2fd2e33..6908faaa500ae 100644 --- a/js/xbtce.js +++ b/js/xbtce.js @@ -16,10 +16,12 @@ module.exports = class xbtce extends Exchange { 'countries': 'RU', 'rateLimit': 2000, // responses are cached every 2 seconds 'version': 'v1', - 'hasPublicAPI': false, - 'hasCORS': false, - 'hasFetchTickers': true, - 'hasFetchOHLCV': false, + 'has': { + 'publicAPI': false, + 'CORS': false, + 'fetchTickers': true, + 'fetchOHLCV': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28059414-e235970c-662c-11e7-8c3a-08e31f78684b.jpg', 'api': 'https://cryptottlivewebapi.xbtce.net:8443/api', diff --git a/js/yobit.js b/js/yobit.js index c7a6e0c141d49..a33db83f0176b 100644 --- a/js/yobit.js +++ b/js/yobit.js @@ -16,8 +16,10 @@ module.exports = class yobit extends liqui { 'countries': 'RU', 'rateLimit': 3000, // responses are cached every 2 seconds 'version': '3', - 'hasCORS': false, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'withdraw': true + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766910-cdcbfdae-5eea-11e7-9859-03fea873272d.jpg', 'api': { diff --git a/js/yunbi.js b/js/yunbi.js index 7bf50d0bd53f6..b4867c5067b57 100644 --- a/js/yunbi.js +++ b/js/yunbi.js @@ -15,9 +15,11 @@ module.exports = class yunbi extends acx { 'countries': 'CN', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': false, - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, + 'has': { + 'CORS': false, + 'fetchTickers': true, + 'fetchOHLCV': true, + }, 'timeframes': { '1m': '1', '5m': '5', diff --git a/js/zaif.js b/js/zaif.js index 9747ea723a323..fc116ccee8310 100644 --- a/js/zaif.js +++ b/js/zaif.js @@ -16,10 +16,12 @@ module.exports = class zaif extends Exchange { 'countries': 'JP', 'rateLimit': 2000, 'version': '1', - 'hasCORS': false, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'fetchOpenOrders': true, + 'fetchClosedOrders': true, + 'withdraw': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766927-39ca2ada-5eeb-11e7-972f-1b4199518ca6.jpg', 'api': 'https://api.zaif.jp', diff --git a/js/zb.js b/js/zb.js index ccdf238fdd54a..9c68afb672550 100644 --- a/js/zb.js +++ b/js/zb.js @@ -16,8 +16,10 @@ module.exports = class zb extends Exchange { 'countries': 'CN', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': false, - 'hasFetchOrder': true, + 'has': { + 'CORS': false, + 'fetchOrder': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/32859187-cd5214f0-ca5e-11e7-967d-96568e2e2bd1.jpg', 'api': { From 00b59f6dbea4c362f6e9173c45600694614d4949 Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 06:56:15 +0300 Subject: [PATCH 31/38] automatic mapping of .has.xxx to hasXXX in constructor, removed a big chunk of bloat (clunky hardcode mapping) + added a test for that --- js/base/Exchange.js | 42 +++++++++++++++--------------------------- js/coinsecure.js | 4 +++- js/test/test_base.js | 19 ++++++++++++++++++- 3 files changed, 36 insertions(+), 29 deletions(-) diff --git a/js/base/Exchange.js b/js/base/Exchange.js index 2783346fc9e8b..d229fe26dcdbb 100644 --- a/js/base/Exchange.js +++ b/js/base/Exchange.js @@ -18,7 +18,8 @@ const { isNode , uuid , unCamelCase , precisionFromString - , throttle } = functions + , throttle + , capitalize } = functions const { sleep , timeout @@ -108,28 +109,6 @@ module.exports = class Exchange { this.twofa = false // two-factor authentication (2FA) this.timeframes = undefined - // TODO: generate - this.hasPublicAPI = true - this.hasPrivateAPI = true - this.hasCORS = false - this.hasDeposit = false - this.hasFetchBalance = true - this.hasFetchClosedOrders = false - this.hasFetchCurrencies = false - this.hasFetchMyTrades = false - this.hasFetchOHLCV = false - this.hasFetchOpenOrders = false - this.hasFetchOrder = false - this.hasFetchOrderBook = true - this.hasFetchOrders = false - this.hasFetchTicker = true - this.hasFetchTickers = false - this.hasFetchBidsAsks = false - this.hasFetchTrades = true - this.hasWithdraw = false - this.hasCreateOrder = this.hasPrivateAPI - this.hasCancelOrder = this.hasPrivateAPI - this.apiKey = undefined this.secret = undefined this.uid = undefined @@ -164,13 +143,17 @@ module.exports = class Exchange { for (const k of names) this[unCamelCase (k)] = this[k] - // API methods metainfo + /* exchange's capabilities (overrideable) */ + this.has = { - 'cancelOrder': this.hasPrivateAPI, + 'CORS': false, + 'publicAPI': true, + 'privateAPI': true, + 'cancelOrder': true, 'createDepositAddress': false, - 'createOrder': this.hasPrivateAPI, + 'createOrder': true, 'deposit': false, - 'fetchBalance': this.hasPrivateAPI, + 'fetchBalance': true, 'fetchClosedOrders': false, 'fetchCurrencies': false, 'fetchDepositAddress': false, @@ -195,6 +178,11 @@ module.exports = class Exchange { for (const [property, value] of Object.entries (config)) this[property] = deepExtend (this[property], value) + // generate old metainfo interface + for (const k in this.has) { + this['has' + capitalize (k)] = !!this.has[k] // converts 'emulated' to true + } + if (this.api) this.defineRestApi (this.api, 'request') diff --git a/js/coinsecure.js b/js/coinsecure.js index 376178431de19..e08142e2bb3c3 100644 --- a/js/coinsecure.js +++ b/js/coinsecure.js @@ -16,7 +16,9 @@ module.exports = class coinsecure extends Exchange { 'countries': 'IN', // India 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766472-9cbd200a-5ed9-11e7-9551-2267ad7bac08.jpg', 'api': 'https://api.coinsecure.in', diff --git a/js/test/test_base.js b/js/test/test_base.js index 9c39f1350e8d3..e525359c49c28 100644 --- a/js/test/test_base.js +++ b/js/test/test_base.js @@ -467,7 +467,7 @@ describe ('ccxt base code', () => { assert.deepEqual (ccxt.sortBy ([], 'x'), []) }) - it.only ('camelCase/camel_case property conversion works', () => { + it ('camelCase/camel_case property conversion works', () => { const exchange = new ccxt.Exchange ({ 'id': 'mock' }) @@ -487,6 +487,23 @@ describe ('ccxt base code', () => { log.magenta.noLocate (`+ ${k}`) }) + + it ('legacy .hasSomething maps to has.something automatically', () => { + + const exchange = new ccxt.Exchange ({ + id: 'mock', + has: { + CORS: true, + publicAPI: false, + fetchDepositAddress: 'emulated' + } + }) + + assert (exchange.hasCORS === true) + assert (exchange.hasPublicAPI === false) + assert (exchange.hasFetchDepositAddress === true) + }) + }) /* ------------------------------------------------------------------------ */ From eccfdb244b5ee10ec8e53281f1ac61c1e20e7034 Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 08:20:46 +0300 Subject: [PATCH 32/38] toPrecision function WIP + tests --- js/base/functions/number.js | 60 ++++++++++++++++++++++++++----------- js/test/test_base.js | 53 ++++++++++++++++++++++---------- 2 files changed, 80 insertions(+), 33 deletions(-) diff --git a/js/base/functions/number.js b/js/base/functions/number.js index 0957400c9eb0a..8955d2974941f 100644 --- a/js/base/functions/number.js +++ b/js/base/functions/number.js @@ -1,16 +1,16 @@ "use strict"; +const { isString, isNumber } = require ('./type') + /* ------------------------------------------------------------------------ */ const decimal = float => parseFloat (float).toString () /* ------------------------------------------------------------------------ */ -//toPrecision (x, { round: true, digits: 6, fixed: true, pad: true, output: 'string' }) - // See https://stackoverflow.com/questions/1685680/how-to-avoid-scientific-notation-for-large-numbers-in-javascript for discussion -function toFixed (x) { // avoid scientific notation for too large and too small numbers +function numberToString (x) { // avoid scientific notation for too large and too small numbers if (Math.abs (x) < 1.0) { const e = parseInt (x.toString ().split ('e-')[1]) @@ -26,41 +26,65 @@ function toFixed (x) { // avoid scientific notation for too large and too small x += (new Array (e+1)).join ('0') } } - return x + return x.toString () } -const numberToString = toFixed - // See https://stackoverflow.com/questions/4912788/truncate-not-round-off-decimal-numbers-in-javascript for discussion // > So, after all it turned out, rounding bugs will always haunt you, no matter how hard you try to compensate them. // > Hence the problem should be attacked by representing numbers exactly in decimal notation. const regexCache = [] - , truncateToString = (num, precision = 0) => { - num = toFixed (num) - if (precision > 0) { - const re = regexCache[precision] || (regexCache[precision] = new RegExp("([-]*\\d+\\.\\d{" + precision + "})(\\d)")) - const [,result] = num.toString ().match (re) || [null, num] - return result.toString () - } - return parseInt (num, 10).toString () + +const padWithZeroes = (x, digits = 0) => { + + const [int, frac = ''] = x.split ('.') + + return int + ((frac || (digits > 0)) + ? ('.' + frac.padEnd (digits, '0')) + : '') +} + +const truncateNumber = (x, { digits = 0, fixed = true }) => { // accepts either strings or Numbers + + const s = isNumber (x) ? numberToString (x) : String (x) + + if (digits > 0) { + const re = regexCache[digits] || (regexCache[digits] = new RegExp("([-]*\\d+\\.\\d{" + digits + "})(\\d)")) + const [,result] = s.match (re) || [null, s] + return fixed + ? padWithZeroes (result, digits) + : result + + } else { + throw new Error ('not implemented yet') } - , truncate = (num, precision = 0) => parseFloat (truncateToString (num, precision)) +} const precisionFromString = (string) => { const split = string.replace (/0+$/g, '').split ('.') return (split.length > 1) ? (split[1].length) : 0 } +const toPrecision = (x, { round = true, digits = 8, fixed = true, output = 'string' }) => { // accepts either strings or Numbers + + const s = !round ? truncateNumber (x, { digits, fixed }) + : (fixed ? Number (x).toFixed (digits) : + Number (x).toPrecision (digits)) + + return (output === 'string') ? s + : Number (s) +} + + /* ------------------------------------------------------------------------ */ module.exports = { decimal, - toFixed, - truncate, - truncateToString, + numberToString, + toPrecision, + padWithZeroes, precisionFromString } diff --git a/js/test/test_base.js b/js/test/test_base.js index e525359c49c28..e836a5f9eaf0e 100644 --- a/js/test/test_base.js +++ b/js/test/test_base.js @@ -22,21 +22,6 @@ const chai = require ('chai') describe ('ccxt base code', () => { - it ('amountToString is robust', async () => { - - assert.strictEqual (ccxt.truncate_to_string (10, 0), '10') - // assert.strictEqual (ccxt.truncate_to_string (10, 1), '10.0') - assert.strictEqual (ccxt.truncate_to_string (10.1, 0), '10') - assert.strictEqual (ccxt.truncate_to_string (10.1, 1), '10.1') - assert.strictEqual (ccxt.truncate_to_string (10.1, 2), '10.1') - assert.strictEqual (ccxt.truncate_to_string (10.11, 2), '10.11') - assert.strictEqual (ccxt.truncate_to_string (10.199, 2), '10.19') - assert.strictEqual (ccxt.truncate_to_string (10.999999, 8), '10.999999') - assert.strictEqual (ccxt.truncate_to_string (10.99999999, 8), '10.99999999') - assert.strictEqual (ccxt.truncate_to_string (10.9999999999, 8), '10.99999999') - - }) - it ('safeFloat/safeInteger is robust', async () => { const $default = {} @@ -504,6 +489,44 @@ describe ('ccxt base code', () => { assert (exchange.hasFetchDepositAddress === true) }) + it.only ('padWithZeros (from number.js) works', () => { + + const { padWithZeroes } = require ('../base/functions/number') + + assert.strictEqual (padWithZeroes ('123.45', -5), '123.45') + assert.strictEqual (padWithZeroes ('123.45', 0), '123.45') + assert.strictEqual (padWithZeroes ('123.45', 1), '123.45') + assert.strictEqual (padWithZeroes ('123', 10), '123.0000000000') + assert.strictEqual (padWithZeroes ('123.4', 10), '123.4000000000') + assert.strictEqual (padWithZeroes ('123.4567', 10), '123.4567000000') + }) + + it.only ('number.js works', () => { + + const { toPrecision, truncate } = require ('../base/functions/number') + + assert.strictEqual (toPrecision (10, { digits: 0 }), '10') + assert.strictEqual (toPrecision (10, { digits: 1 }), '10.0') + assert.strictEqual (toPrecision (10.1, { digits: 0 }), '10') + assert.strictEqual (toPrecision (10.1, { digits: 1 }), '10.1') + + assert.strictEqual (toPrecision (10.1, { digits: 2, round: false }), '10.10') + assert.strictEqual (toPrecision (10.11, { digits: 2, round: false }), '10.11') + assert.strictEqual (toPrecision (10.199, { digits: 2, round: false }), '10.19') + assert.strictEqual (toPrecision (10.999999, { digits: 8, round: false }), '10.99999900') + assert.strictEqual (toPrecision (10.99999999, { digits: 8, round: false }), '10.99999999') + assert.strictEqual (toPrecision (10.9999999999, { digits: 8, round: false }), '10.99999999') + + assert.strictEqual (toPrecision (123, { round: false, digits: 4 }), '123.0000') + assert.strictEqual (toPrecision (123.49999999, { round: false, digits: 4 }), '123.4999') + assert.strictEqual (toPrecision (123.49999999, { round: true, digits: 4 }), '123.5000') + assert.strictEqual (toPrecision (123.5, { round: false, digits: 4 }), '123.5000') + + assert.strictEqual (toPrecision (123.49999999, { round: false, digits: 4, output: 'number' }), 123.4999) + assert.strictEqual (toPrecision (123.49999999, { round: true, digits: 4, output: 'number' }), 123.5) + assert.strictEqual (toPrecision (123.5, { round: false, digits: 4, output: 'number' }), 123.5) + }) + }) /* ------------------------------------------------------------------------ */ From 9b1b49bb24015bf901f2f9c4edca9a7718b19b6c Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 13:03:40 +0300 Subject: [PATCH 33/38] working on the general-purpose toPrecision function, roundDecimalString WIP (poor man's Decimal.js, haha) --- js/base/functions/number.js | 57 ++++++++++++++++++++------ js/test/test_base.js | 79 ++++++++++++++++++++++++++----------- 2 files changed, 103 insertions(+), 33 deletions(-) diff --git a/js/base/functions/number.js b/js/base/functions/number.js index 8955d2974941f..f38958f426da4 100644 --- a/js/base/functions/number.js +++ b/js/base/functions/number.js @@ -29,13 +29,6 @@ function numberToString (x) { // avoid scientific notation for too large and too return x.toString () } -// See https://stackoverflow.com/questions/4912788/truncate-not-round-off-decimal-numbers-in-javascript for discussion - -// > So, after all it turned out, rounding bugs will always haunt you, no matter how hard you try to compensate them. -// > Hence the problem should be attacked by representing numbers exactly in decimal notation. - -const regexCache = [] - const padWithZeroes = (x, digits = 0) => { const [int, frac = ''] = x.split ('.') @@ -45,7 +38,49 @@ const padWithZeroes = (x, digits = 0) => { : '') } -const truncateNumber = (x, { digits = 0, fixed = true }) => { // accepts either strings or Numbers +const roundDecimalString = (s, to, afterDot = false) => { + + const digits = Array.from (s) + const result = [] + + let memo = 0 + let remaining = digits.length + + const dot = s.indexOf ('.') + if (afterDot) to = dot + to + + for (let i = digits.length - 1; i >= 0; i--) { + const d = digits[i] + if (d !== '.') { + let n = (d.charCodeAt (0) - 48) + memo + let numDigitsAhead = i + let dotAhead = (dot >= 0) && (i >= dot) + if ((numDigitsAhead + (dotAhead ? -1 : 0)) >= to) { // ignore dot when counting digits ahead + n = (n > 5) ? 10 : 0 // rounding on per-digit basis + } + if (n > 9) { n = 0; memo = 1; } + else memo = 0 + digits[i] = n + } + } + return (memo || '') + digits.join ('') +} + + +const roundNumber = (x, { digits = 8, fixed = true }) => { + + const [,zeros,significantPart] = numberToString (x).match (/^([^1-9]*)(.+)$/) + + return zeros + roundDecimalString (significantPart, digits, fixed) +} + +// See https://stackoverflow.com/questions/4912788/truncate-not-round-off-decimal-numbers-in-javascript for discussion + +// > So, after all it turned out, rounding bugs will always haunt you, no matter how hard you try to compensate them. +// > Hence the problem should be attacked by representing numbers exactly in decimal notation. + +const regexCache = [] +const truncNumber = (x, { digits = 0, fixed = true }) => { // accepts either strings or Numbers const s = isNumber (x) ? numberToString (x) : String (x) @@ -68,9 +103,8 @@ const precisionFromString = (string) => { const toPrecision = (x, { round = true, digits = 8, fixed = true, output = 'string' }) => { // accepts either strings or Numbers - const s = !round ? truncateNumber (x, { digits, fixed }) - : (fixed ? Number (x).toFixed (digits) : - Number (x).toPrecision (digits)) + const s = round ? roundNumber (x, { digits, fixed }) + : truncNumber (x, { digits, fixed }) return (output === 'string') ? s : Number (s) @@ -85,6 +119,7 @@ module.exports = { numberToString, toPrecision, padWithZeroes, + roundDecimalString, precisionFromString } diff --git a/js/test/test_base.js b/js/test/test_base.js index e836a5f9eaf0e..17c5c47f82647 100644 --- a/js/test/test_base.js +++ b/js/test/test_base.js @@ -503,28 +503,63 @@ describe ('ccxt base code', () => { it.only ('number.js works', () => { - const { toPrecision, truncate } = require ('../base/functions/number') - - assert.strictEqual (toPrecision (10, { digits: 0 }), '10') - assert.strictEqual (toPrecision (10, { digits: 1 }), '10.0') - assert.strictEqual (toPrecision (10.1, { digits: 0 }), '10') - assert.strictEqual (toPrecision (10.1, { digits: 1 }), '10.1') - - assert.strictEqual (toPrecision (10.1, { digits: 2, round: false }), '10.10') - assert.strictEqual (toPrecision (10.11, { digits: 2, round: false }), '10.11') - assert.strictEqual (toPrecision (10.199, { digits: 2, round: false }), '10.19') - assert.strictEqual (toPrecision (10.999999, { digits: 8, round: false }), '10.99999900') - assert.strictEqual (toPrecision (10.99999999, { digits: 8, round: false }), '10.99999999') - assert.strictEqual (toPrecision (10.9999999999, { digits: 8, round: false }), '10.99999999') - - assert.strictEqual (toPrecision (123, { round: false, digits: 4 }), '123.0000') - assert.strictEqual (toPrecision (123.49999999, { round: false, digits: 4 }), '123.4999') - assert.strictEqual (toPrecision (123.49999999, { round: true, digits: 4 }), '123.5000') - assert.strictEqual (toPrecision (123.5, { round: false, digits: 4 }), '123.5000') - - assert.strictEqual (toPrecision (123.49999999, { round: false, digits: 4, output: 'number' }), 123.4999) - assert.strictEqual (toPrecision (123.49999999, { round: true, digits: 4, output: 'number' }), 123.5) - assert.strictEqual (toPrecision (123.5, { round: false, digits: 4, output: 'number' }), 123.5) + const { toPrecision, truncate, roundDecimalString } = require ('../base/functions/number') + + assert.strictEqual (roundDecimalString ('1234567890', 100), '1234567890') + assert.strictEqual (roundDecimalString ('1234567890', 10), '1234567890') + assert.strictEqual (roundDecimalString ('1234567890', 8), '1234567900') + assert.strictEqual (roundDecimalString ('1234567890', 7), '1234568000') + assert.strictEqual (roundDecimalString ('1234567890', 6), '1234570000') + assert.strictEqual (roundDecimalString ('1234567890', 5), '1234600000') + assert.strictEqual (roundDecimalString ('1234567890', 4), '1235000000') + assert.strictEqual (roundDecimalString ('1234567890', 3), '1230000000') + assert.strictEqual (roundDecimalString ('1234567890', 2), '1200000000') + assert.strictEqual (roundDecimalString ('1234567890', 1), '1000000000') + assert.strictEqual (roundDecimalString ('1234567890', 0), '0000000000') + + assert.strictEqual (roundDecimalString ('9999999999', 0), '10000000000') + assert.strictEqual (roundDecimalString ('9999.999999', 0), '10000.000000') + + assert.strictEqual (roundDecimalString ('1234.567890', 7), '1234.568000') + assert.strictEqual (roundDecimalString ('12345.67890', 7), '12345.68000') + assert.strictEqual (roundDecimalString ('123456.7890', 7), '123456.8000') + assert.strictEqual (roundDecimalString ('1234567.890', 7), '1234568.000') + + + assert.strictEqual (roundDecimalString ('1234.567890', 2, true), '1234.570000') // after dot + assert.strictEqual (roundDecimalString ('1234.567890', 0, true), '1235.000000') // after dot + + return + + // assert.strictEqual (toPrecision (10, { digits: 0 }), '10') + // assert.strictEqual (toPrecision (10, { digits: 1 }), '10.0') + // assert.strictEqual (toPrecision (10.1, { digits: 0 }), '10') + // assert.strictEqual (toPrecision (10.1, { digits: 1 }), '10.1') + + // assert.strictEqual (toPrecision (10.1, { digits: 2, round: false }), '10.10') + // assert.strictEqual (toPrecision (10.11, { digits: 2, round: false }), '10.11') + // assert.strictEqual (toPrecision (10.199, { digits: 2, round: false }), '10.19') + // assert.strictEqual (toPrecision (10.999999, { digits: 8, round: false }), '10.99999900') + // assert.strictEqual (toPrecision (10.99999999, { digits: 8, round: false }), '10.99999999') + // assert.strictEqual (toPrecision (10.9999999999, { digits: 8, round: false }), '10.99999999') + + // assert.strictEqual (toPrecision (123, { round: false, digits: 4 }), '123.0000') + // assert.strictEqual (toPrecision (123.49999999, { round: false, digits: 4 }), '123.4999') + // assert.strictEqual (toPrecision (123.49999999, { round: true, digits: 4 }), '123.5000') + // assert.strictEqual (toPrecision (123.5, { round: false, digits: 4 }), '123.5000') + + // assert.strictEqual (toPrecision (123.49999999, { round: false, digits: 4, output: 'number' }), 123.4999) + // assert.strictEqual (toPrecision (123.49999999, { round: true, digits: 4, output: 'number' }), 123.5) + // assert.strictEqual (toPrecision (123.5, { round: false, digits: 4, output: 'number' }), 123.5) + + // assert.strictEqual (toPrecision (123.000789, { digits: 8, fixed: false }), '123.00079') // due to rounding + // assert.strictEqual (toPrecision (123.000789, { digits: 8, fixed: false, round: false }), '123.00078') // no rounding + + + //assert.strictEqual (toPrecision (0.000000012345678, { digits: 8, fixed: false }), '0.000000012345678') + assert.strictEqual (toPrecision (0.000000012345678, { digits: 5, fixed: false }), '0.00000001234567') + + }) }) From dff044659cd50f744189b8538b841ffb00e8532f Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 13:23:37 +0300 Subject: [PATCH 34/38] roundNumber WIP (still not working yet) --- js/base/functions/number.js | 21 ++++++++++++------ js/test/test_base.js | 43 ++++++++++++++++++------------------- 2 files changed, 35 insertions(+), 29 deletions(-) diff --git a/js/base/functions/number.js b/js/base/functions/number.js index f38958f426da4..d5e4cd95975ca 100644 --- a/js/base/functions/number.js +++ b/js/base/functions/number.js @@ -12,6 +12,8 @@ const decimal = float => parseFloat (float).toString () function numberToString (x) { // avoid scientific notation for too large and too small numbers + if (isString (x)) return x + if (Math.abs (x) < 1.0) { const e = parseInt (x.toString ().split ('e-')[1]) if (e) { @@ -42,12 +44,11 @@ const roundDecimalString = (s, to, afterDot = false) => { const digits = Array.from (s) const result = [] + const dot = s.indexOf ('.') let memo = 0 - let remaining = digits.length - const dot = s.indexOf ('.') - if (afterDot) to = dot + to + if (afterDot) to = ((dot >= 0) ? dot : digits.length) + to for (let i = digits.length - 1; i >= 0; i--) { const d = digits[i] @@ -67,11 +68,17 @@ const roundDecimalString = (s, to, afterDot = false) => { } -const roundNumber = (x, { digits = 8, fixed = true }) => { +const roundNumber = (x, { digits = 8, fixed = true }) => { // accepts either strings or Numbers - const [,zeros,significantPart] = numberToString (x).match (/^([^1-9]*)(.+)$/) + const s = numberToString (x) - return zeros + roundDecimalString (significantPart, digits, fixed) + if (fixed) { + return roundDecimalString (s, digits, true) + + } else { + const [,zeros,significantPart] = s.match (/^([^1-9]*)(.+)$/) + return zeros + roundDecimalString (significantPart, digits) + } } // See https://stackoverflow.com/questions/4912788/truncate-not-round-off-decimal-numbers-in-javascript for discussion @@ -82,7 +89,7 @@ const roundNumber = (x, { digits = 8, fixed = true }) => { const regexCache = [] const truncNumber = (x, { digits = 0, fixed = true }) => { // accepts either strings or Numbers - const s = isNumber (x) ? numberToString (x) : String (x) + const s = numberToString (x) if (digits > 0) { const re = regexCache[digits] || (regexCache[digits] = new RegExp("([-]*\\d+\\.\\d{" + digits + "})(\\d)")) diff --git a/js/test/test_base.js b/js/test/test_base.js index 17c5c47f82647..2703954a3a622 100644 --- a/js/test/test_base.js +++ b/js/test/test_base.js @@ -529,35 +529,34 @@ describe ('ccxt base code', () => { assert.strictEqual (roundDecimalString ('1234.567890', 2, true), '1234.570000') // after dot assert.strictEqual (roundDecimalString ('1234.567890', 0, true), '1235.000000') // after dot - return + assert.strictEqual (toPrecision (10, { digits: 0 }), '10') + assert.strictEqual (toPrecision (10, { digits: 1 }), '10.0') + assert.strictEqual (toPrecision (10.1, { digits: 0 }), '10') + assert.strictEqual (toPrecision (10.1, { digits: 1 }), '10.1') - // assert.strictEqual (toPrecision (10, { digits: 0 }), '10') - // assert.strictEqual (toPrecision (10, { digits: 1 }), '10.0') - // assert.strictEqual (toPrecision (10.1, { digits: 0 }), '10') - // assert.strictEqual (toPrecision (10.1, { digits: 1 }), '10.1') + assert.strictEqual (toPrecision (10.1, { digits: 2, round: false }), '10.10') + assert.strictEqual (toPrecision (10.11, { digits: 2, round: false }), '10.11') + assert.strictEqual (toPrecision (10.199, { digits: 2, round: false }), '10.19') + assert.strictEqual (toPrecision (10.999999, { digits: 8, round: false }), '10.99999900') + assert.strictEqual (toPrecision (10.99999999, { digits: 8, round: false }), '10.99999999') + assert.strictEqual (toPrecision (10.9999999999, { digits: 8, round: false }), '10.99999999') - // assert.strictEqual (toPrecision (10.1, { digits: 2, round: false }), '10.10') - // assert.strictEqual (toPrecision (10.11, { digits: 2, round: false }), '10.11') - // assert.strictEqual (toPrecision (10.199, { digits: 2, round: false }), '10.19') - // assert.strictEqual (toPrecision (10.999999, { digits: 8, round: false }), '10.99999900') - // assert.strictEqual (toPrecision (10.99999999, { digits: 8, round: false }), '10.99999999') - // assert.strictEqual (toPrecision (10.9999999999, { digits: 8, round: false }), '10.99999999') + assert.strictEqual (toPrecision (123, { round: false, digits: 4 }), '123.0000') + assert.strictEqual (toPrecision (123.49999999, { round: false, digits: 4 }), '123.4999') + assert.strictEqual (toPrecision (123.49999999, { round: true, digits: 4 }), '123.5000') + assert.strictEqual (toPrecision (123.5, { round: false, digits: 4 }), '123.5000') - // assert.strictEqual (toPrecision (123, { round: false, digits: 4 }), '123.0000') - // assert.strictEqual (toPrecision (123.49999999, { round: false, digits: 4 }), '123.4999') - // assert.strictEqual (toPrecision (123.49999999, { round: true, digits: 4 }), '123.5000') - // assert.strictEqual (toPrecision (123.5, { round: false, digits: 4 }), '123.5000') + assert.strictEqual (toPrecision (123.49999999, { round: false, digits: 4, output: 'number' }), 123.4999) + assert.strictEqual (toPrecision (123.49999999, { round: true, digits: 4, output: 'number' }), 123.5) + assert.strictEqual (toPrecision (123.5, { round: false, digits: 4, output: 'number' }), 123.5) - // assert.strictEqual (toPrecision (123.49999999, { round: false, digits: 4, output: 'number' }), 123.4999) - // assert.strictEqual (toPrecision (123.49999999, { round: true, digits: 4, output: 'number' }), 123.5) - // assert.strictEqual (toPrecision (123.5, { round: false, digits: 4, output: 'number' }), 123.5) + assert.strictEqual (toPrecision (123.000789, { digits: 8, fixed: false }), '123.00079') // due to rounding + assert.strictEqual (toPrecision (123.000789, { digits: 8, fixed: false, round: false }), '123.00078') // no rounding - // assert.strictEqual (toPrecision (123.000789, { digits: 8, fixed: false }), '123.00079') // due to rounding - // assert.strictEqual (toPrecision (123.000789, { digits: 8, fixed: false, round: false }), '123.00078') // no rounding - - //assert.strictEqual (toPrecision (0.000000012345678, { digits: 8, fixed: false }), '0.000000012345678') + assert.strictEqual (toPrecision (0.000000012345678, { digits: 8, fixed: false }), '0.000000012345678') assert.strictEqual (toPrecision (0.000000012345678, { digits: 5, fixed: false }), '0.00000001234567') + assert.strictEqual (toPrecision (0.000000012345678, { digits: 3, fixed: false }), '0.00000001234567') }) From f0d9ba581dd90fc27664ac077d2871aa2332c5e1 Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 13:26:30 +0300 Subject: [PATCH 35/38] wip --- js/base/functions/number.js | 20 ++++++++++++-------- 1 file changed, 12 insertions(+), 8 deletions(-) diff --git a/js/base/functions/number.js b/js/base/functions/number.js index d5e4cd95975ca..df3d67870db99 100644 --- a/js/base/functions/number.js +++ b/js/base/functions/number.js @@ -31,6 +31,8 @@ function numberToString (x) { // avoid scientific notation for too large and too return x.toString () } +/* ------------------------------------------------------------------------ */ + const padWithZeroes = (x, digits = 0) => { const [int, frac = ''] = x.split ('.') @@ -40,6 +42,8 @@ const padWithZeroes = (x, digits = 0) => { : '') } +/* ------------------------------------------------------------------------ */ + const roundDecimalString = (s, to, afterDot = false) => { const digits = Array.from (s) @@ -67,6 +71,7 @@ const roundDecimalString = (s, to, afterDot = false) => { return (memo || '') + digits.join ('') } +/* ------------------------------------------------------------------------ */ const roundNumber = (x, { digits = 8, fixed = true }) => { // accepts either strings or Numbers @@ -81,6 +86,8 @@ const roundNumber = (x, { digits = 8, fixed = true }) => { // accepts either str } } +/* ------------------------------------------------------------------------ */ + // See https://stackoverflow.com/questions/4912788/truncate-not-round-off-decimal-numbers-in-javascript for discussion // > So, after all it turned out, rounding bugs will always haunt you, no matter how hard you try to compensate them. @@ -103,20 +110,17 @@ const truncNumber = (x, { digits = 0, fixed = true }) => { // accepts either str } } +/* ------------------------------------------------------------------------ */ + const precisionFromString = (string) => { const split = string.replace (/0+$/g, '').split ('.') return (split.length > 1) ? (split[1].length) : 0 } -const toPrecision = (x, { round = true, digits = 8, fixed = true, output = 'string' }) => { // accepts either strings or Numbers - - const s = round ? roundNumber (x, { digits, fixed }) - : truncNumber (x, { digits, fixed }) - - return (output === 'string') ? s - : Number (s) -} +/* ------------------------------------------------------------------------ */ +const toPrecision = (x, { round = true, digits = 8, fixed = true }) => round ? roundNumber (x, { digits, fixed }) + : truncNumber (x, { digits, fixed }) /* ------------------------------------------------------------------------ */ From 68c135e6981197df2c2bf84988943edf4a6fde07 Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 13:29:36 +0300 Subject: [PATCH 36/38] fixed a couple of broken/misleading test cases --- js/test/test_base.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/js/test/test_base.js b/js/test/test_base.js index 2703954a3a622..01f34ab713dd0 100644 --- a/js/test/test_base.js +++ b/js/test/test_base.js @@ -555,8 +555,8 @@ describe ('ccxt base code', () => { assert.strictEqual (toPrecision (0.000000012345678, { digits: 8, fixed: false }), '0.000000012345678') - assert.strictEqual (toPrecision (0.000000012345678, { digits: 5, fixed: false }), '0.00000001234567') - assert.strictEqual (toPrecision (0.000000012345678, { digits: 3, fixed: false }), '0.00000001234567') + assert.strictEqual (toPrecision (0.000000012345678, { digits: 5, fixed: false }), '0.000000012345') + assert.strictEqual (toPrecision (0.000000012345678, { digits: 3, fixed: false }), '0.0000000123') }) From a2064cd1bf0fc45605921d7075e498634cc7bcdd Mon Sep 17 00:00:00 2001 From: xpl Date: Wed, 17 Jan 2018 13:33:05 +0300 Subject: [PATCH 37/38] fixed a couple of broken/misleading test cases --- js/test/test_base.js | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/js/test/test_base.js b/js/test/test_base.js index 01f34ab713dd0..c70795400b6f2 100644 --- a/js/test/test_base.js +++ b/js/test/test_base.js @@ -554,9 +554,9 @@ describe ('ccxt base code', () => { assert.strictEqual (toPrecision (123.000789, { digits: 8, fixed: false, round: false }), '123.00078') // no rounding - assert.strictEqual (toPrecision (0.000000012345678, { digits: 8, fixed: false }), '0.000000012345678') - assert.strictEqual (toPrecision (0.000000012345678, { digits: 5, fixed: false }), '0.000000012345') - assert.strictEqual (toPrecision (0.000000012345678, { digits: 3, fixed: false }), '0.0000000123') + assert.strictEqual (toPrecision ('0.000000012345678', { digits: 8, fixed: false }), '0.000000012345678') + assert.strictEqual (toPrecision ('0.000000012345678', { digits: 5, fixed: false }), '0.000000012346') // should round here + assert.strictEqual (toPrecision ('0.000000012345678', { digits: 3, fixed: false }), '0.0000000123') }) From 28d611f2bb9efb23d779ecf1bca84542e8c4cc4b Mon Sep 17 00:00:00 2001 From: xpl Date: Sun, 21 Jan 2018 09:22:02 +0300 Subject: [PATCH 38/38] removed old metainfo interface for python and PHP, fixed failing tests, temporarily returned legacy number.js implementation (to be able to merge current work with the master branch and then continue re-working it) --- build/ccxt.browser.js | 2141 ++++++++++--------- doc/manual.rst | 10 +- examples/js/fetch-ticker-where-available.js | 2 +- examples/js/search-all-exchanges.js | 2 +- js/base/Exchange.js | 19 +- js/base/functions.js | 46 +- js/base/functions/encode.js | 14 +- js/base/functions/misc.js | 22 + js/base/functions/number.js | 230 +- js/base/functions/throttle.js | 10 +- js/base/functions/time.js | 14 +- js/okcoinusd.js | 4 +- js/test/test.js | 4 +- js/test/test_base.js | 23 +- package.json | 4 +- php/Exchange.php | 27 +- php/_1broker.php | 10 +- php/_1btcxe.php | 8 +- php/acx.php | 10 +- php/allcoin.php | 4 +- php/anxpro.php | 8 +- php/bibox.php | 11 +- php/binance.php | 11 +- php/bit2c.php | 4 +- php/bitbay.php | 6 +- php/bitcoincoid.php | 13 +- php/bitfinex.php | 10 +- php/bitfinex2.php | 11 +- php/bitflyer.php | 6 +- php/bithumb.php | 6 +- php/bitlish.php | 10 +- php/bitmarket.php | 8 +- php/bitmex.php | 8 +- php/bitso.php | 4 +- php/bitstamp.php | 6 +- php/bitstamp1.php | 4 +- php/bittrex.php | 12 +- php/bl3p.php | 4 +- php/bleutrade.php | 8 +- php/braziliex.php | 5 - php/btcbox.php | 6 +- php/btcchina.php | 4 +- php/btcexchange.php | 4 +- php/btcmarkets.php | 7 +- php/btctradeua.php | 4 +- php/btcturk.php | 8 +- php/btcx.php | 4 +- php/bter.php | 5 - php/bxinth.php | 6 +- php/ccex.php | 6 +- php/cex.php | 10 +- php/chbtc.php | 6 +- php/chilebit.php | 4 +- php/coincheck.php | 4 +- php/coinexchange.php | 6 +- php/coinfloor.php | 4 +- php/coingi.php | 6 +- php/coinmarketcap.php | 18 +- php/coinmate.php | 4 +- php/coinsecure.php | 4 +- php/coinspot.php | 4 +- php/cryptopia.php | 13 +- php/dsx.php | 16 +- php/exmo.php | 8 +- php/flowbtc.php | 4 +- php/foxbit.php | 4 +- php/fybse.php | 4 +- php/fybsg.php | 4 +- php/gatecoin.php | 8 +- php/gdax.php | 10 - php/gemini.php | 4 - php/hitbtc.php | 13 +- php/hitbtc2.php | 13 +- php/huobi.php | 6 +- php/huobicny.php | 4 +- php/huobipro.php | 7 +- php/independentreserve.php | 4 +- php/itbit.php | 4 +- php/jubi.php | 6 +- php/kraken.php | 12 +- php/kucoin.php | 13 +- php/kuna.php | 8 +- php/lakebtc.php | 4 +- php/liqui.php | 11 +- php/livecoin.php | 6 +- php/luno.php | 4 +- php/lykke.php | 6 +- php/mercado.php | 6 +- php/mixcoins.php | 4 +- php/nova.php | 4 +- php/okcoincny.php | 4 +- php/okcoinusd.php | 14 +- php/okex.php | 6 +- php/paymium.php | 4 +- php/poloniex.php | 13 +- php/qryptos.php | 4 +- php/quadrigacx.php | 5 +- php/quoinex.php | 6 +- php/southxchange.php | 8 +- php/surbitcoin.php | 4 +- php/test/test.php | 2 +- php/therock.php | 6 +- php/tidex.php | 6 +- php/urdubit.php | 4 +- php/vaultoro.php | 4 +- php/vbtc.php | 4 +- php/virwox.php | 4 +- php/wex.php | 6 +- php/xbtce.php | 10 +- php/yobit.php | 6 +- php/yunbi.php | 8 +- php/zaif.php | 10 +- php/zb.php | 6 +- python/ccxt/_1broker.py | 10 +- python/ccxt/_1btcxe.py | 8 +- python/ccxt/acx.py | 10 +- python/ccxt/allcoin.py | 4 +- python/ccxt/anxpro.py | 8 +- python/ccxt/async/_1broker.py | 10 +- python/ccxt/async/_1btcxe.py | 8 +- python/ccxt/async/acx.py | 10 +- python/ccxt/async/allcoin.py | 4 +- python/ccxt/async/anxpro.py | 8 +- python/ccxt/async/bibox.py | 11 +- python/ccxt/async/binance.py | 11 +- python/ccxt/async/bit2c.py | 4 +- python/ccxt/async/bitbay.py | 6 +- python/ccxt/async/bitcoincoid.py | 13 +- python/ccxt/async/bitfinex.py | 10 +- python/ccxt/async/bitfinex2.py | 11 +- python/ccxt/async/bitflyer.py | 6 +- python/ccxt/async/bithumb.py | 6 +- python/ccxt/async/bitlish.py | 10 +- python/ccxt/async/bitmarket.py | 8 +- python/ccxt/async/bitmex.py | 8 +- python/ccxt/async/bitso.py | 4 +- python/ccxt/async/bitstamp.py | 6 +- python/ccxt/async/bitstamp1.py | 4 +- python/ccxt/async/bittrex.py | 12 +- python/ccxt/async/bl3p.py | 4 +- python/ccxt/async/bleutrade.py | 8 +- python/ccxt/async/braziliex.py | 5 - python/ccxt/async/btcbox.py | 6 +- python/ccxt/async/btcchina.py | 4 +- python/ccxt/async/btcexchange.py | 4 +- python/ccxt/async/btcmarkets.py | 7 +- python/ccxt/async/btctradeua.py | 4 +- python/ccxt/async/btcturk.py | 8 +- python/ccxt/async/btcx.py | 4 +- python/ccxt/async/bter.py | 5 - python/ccxt/async/bxinth.py | 6 +- python/ccxt/async/ccex.py | 6 +- python/ccxt/async/cex.py | 10 +- python/ccxt/async/chbtc.py | 6 +- python/ccxt/async/chilebit.py | 4 +- python/ccxt/async/coincheck.py | 4 +- python/ccxt/async/coinexchange.py | 6 +- python/ccxt/async/coinfloor.py | 4 +- python/ccxt/async/coingi.py | 6 +- python/ccxt/async/coinmarketcap.py | 18 +- python/ccxt/async/coinmate.py | 4 +- python/ccxt/async/coinsecure.py | 4 +- python/ccxt/async/coinspot.py | 4 +- python/ccxt/async/cryptopia.py | 13 +- python/ccxt/async/dsx.py | 16 +- python/ccxt/async/exmo.py | 8 +- python/ccxt/async/flowbtc.py | 4 +- python/ccxt/async/foxbit.py | 4 +- python/ccxt/async/fybse.py | 4 +- python/ccxt/async/fybsg.py | 4 +- python/ccxt/async/gatecoin.py | 8 +- python/ccxt/async/gdax.py | 10 - python/ccxt/async/gemini.py | 4 - python/ccxt/async/hitbtc.py | 13 +- python/ccxt/async/hitbtc2.py | 13 +- python/ccxt/async/huobi.py | 6 +- python/ccxt/async/huobicny.py | 4 +- python/ccxt/async/huobipro.py | 7 +- python/ccxt/async/independentreserve.py | 4 +- python/ccxt/async/itbit.py | 4 +- python/ccxt/async/jubi.py | 6 +- python/ccxt/async/kraken.py | 12 +- python/ccxt/async/kucoin.py | 13 +- python/ccxt/async/kuna.py | 8 +- python/ccxt/async/lakebtc.py | 4 +- python/ccxt/async/liqui.py | 11 +- python/ccxt/async/livecoin.py | 6 +- python/ccxt/async/luno.py | 4 +- python/ccxt/async/lykke.py | 6 +- python/ccxt/async/mercado.py | 6 +- python/ccxt/async/mixcoins.py | 4 +- python/ccxt/async/nova.py | 4 +- python/ccxt/async/okcoincny.py | 4 +- python/ccxt/async/okcoinusd.py | 14 +- python/ccxt/async/okex.py | 6 +- python/ccxt/async/paymium.py | 4 +- python/ccxt/async/poloniex.py | 13 +- python/ccxt/async/qryptos.py | 4 +- python/ccxt/async/quadrigacx.py | 5 +- python/ccxt/async/quoinex.py | 6 +- python/ccxt/async/southxchange.py | 8 +- python/ccxt/async/surbitcoin.py | 4 +- python/ccxt/async/therock.py | 6 +- python/ccxt/async/tidex.py | 6 +- python/ccxt/async/urdubit.py | 4 +- python/ccxt/async/vaultoro.py | 4 +- python/ccxt/async/vbtc.py | 4 +- python/ccxt/async/virwox.py | 4 +- python/ccxt/async/wex.py | 6 +- python/ccxt/async/xbtce.py | 10 +- python/ccxt/async/yobit.py | 6 +- python/ccxt/async/yunbi.py | 8 +- python/ccxt/async/zaif.py | 10 +- python/ccxt/async/zb.py | 6 +- python/ccxt/base/exchange.py | 27 +- python/ccxt/bibox.py | 11 +- python/ccxt/binance.py | 11 +- python/ccxt/bit2c.py | 4 +- python/ccxt/bitbay.py | 6 +- python/ccxt/bitcoincoid.py | 13 +- python/ccxt/bitfinex.py | 10 +- python/ccxt/bitfinex2.py | 11 +- python/ccxt/bitflyer.py | 6 +- python/ccxt/bithumb.py | 6 +- python/ccxt/bitlish.py | 10 +- python/ccxt/bitmarket.py | 8 +- python/ccxt/bitmex.py | 8 +- python/ccxt/bitso.py | 4 +- python/ccxt/bitstamp.py | 6 +- python/ccxt/bitstamp1.py | 4 +- python/ccxt/bittrex.py | 12 +- python/ccxt/bl3p.py | 4 +- python/ccxt/bleutrade.py | 8 +- python/ccxt/braziliex.py | 5 - python/ccxt/btcbox.py | 6 +- python/ccxt/btcchina.py | 4 +- python/ccxt/btcexchange.py | 4 +- python/ccxt/btcmarkets.py | 7 +- python/ccxt/btctradeua.py | 4 +- python/ccxt/btcturk.py | 8 +- python/ccxt/btcx.py | 4 +- python/ccxt/bter.py | 5 - python/ccxt/bxinth.py | 6 +- python/ccxt/ccex.py | 6 +- python/ccxt/cex.py | 10 +- python/ccxt/chbtc.py | 6 +- python/ccxt/chilebit.py | 4 +- python/ccxt/coincheck.py | 4 +- python/ccxt/coinexchange.py | 6 +- python/ccxt/coinfloor.py | 4 +- python/ccxt/coingi.py | 6 +- python/ccxt/coinmarketcap.py | 18 +- python/ccxt/coinmate.py | 4 +- python/ccxt/coinsecure.py | 4 +- python/ccxt/coinspot.py | 4 +- python/ccxt/cryptopia.py | 13 +- python/ccxt/dsx.py | 16 +- python/ccxt/exmo.py | 8 +- python/ccxt/flowbtc.py | 4 +- python/ccxt/foxbit.py | 4 +- python/ccxt/fybse.py | 4 +- python/ccxt/fybsg.py | 4 +- python/ccxt/gatecoin.py | 8 +- python/ccxt/gdax.py | 10 - python/ccxt/gemini.py | 4 - python/ccxt/hitbtc.py | 13 +- python/ccxt/hitbtc2.py | 13 +- python/ccxt/huobi.py | 6 +- python/ccxt/huobicny.py | 4 +- python/ccxt/huobipro.py | 7 +- python/ccxt/independentreserve.py | 4 +- python/ccxt/itbit.py | 4 +- python/ccxt/jubi.py | 6 +- python/ccxt/kraken.py | 12 +- python/ccxt/kucoin.py | 13 +- python/ccxt/kuna.py | 8 +- python/ccxt/lakebtc.py | 4 +- python/ccxt/liqui.py | 11 +- python/ccxt/livecoin.py | 6 +- python/ccxt/luno.py | 4 +- python/ccxt/lykke.py | 6 +- python/ccxt/mercado.py | 6 +- python/ccxt/mixcoins.py | 4 +- python/ccxt/nova.py | 4 +- python/ccxt/okcoincny.py | 4 +- python/ccxt/okcoinusd.py | 14 +- python/ccxt/okex.py | 6 +- python/ccxt/paymium.py | 4 +- python/ccxt/poloniex.py | 13 +- python/ccxt/qryptos.py | 4 +- python/ccxt/quadrigacx.py | 5 +- python/ccxt/quoinex.py | 6 +- python/ccxt/southxchange.py | 8 +- python/ccxt/surbitcoin.py | 4 +- python/ccxt/therock.py | 6 +- python/ccxt/tidex.py | 6 +- python/ccxt/urdubit.py | 4 +- python/ccxt/vaultoro.py | 4 +- python/ccxt/vbtc.py | 4 +- python/ccxt/virwox.py | 4 +- python/ccxt/wex.py | 6 +- python/ccxt/xbtce.py | 10 +- python/ccxt/yobit.py | 6 +- python/ccxt/yunbi.py | 8 +- python/ccxt/zaif.py | 10 +- python/ccxt/zb.py | 6 +- python/test/test.py | 12 +- python/test/test_async.py | 12 +- run-tests.js | 2 - wiki/Manual.md | 35 +- 310 files changed, 2325 insertions(+), 2326 deletions(-) create mode 100644 js/base/functions/misc.js diff --git a/build/ccxt.browser.js b/build/ccxt.browser.js index 114de05b4f1ad..e442e8532a7c3 100644 --- a/build/ccxt.browser.js +++ b/build/ccxt.browser.js @@ -39,8 +39,8 @@ SOFTWARE. //----------------------------------------------------------------------------- const Exchange = require ('./js/base/Exchange') -const functions = require ('./js/base/functions') -const errors = require ('./js/base/errors') + , functions = require ('./js/base/functions') + , errors = require ('./js/base/errors') //----------------------------------------------------------------------------- // this is updated by vss.js when building @@ -158,7 +158,7 @@ module.exports = Object.assign ({ version, Exchange, exchanges: Object.keys (exc //----------------------------------------------------------------------------- -},{"./js/_1broker.js":3,"./js/_1btcxe.js":4,"./js/acx.js":5,"./js/allcoin.js":6,"./js/anxpro.js":7,"./js/base/Exchange":8,"./js/base/errors":10,"./js/base/functions":11,"./js/bibox.js":13,"./js/binance.js":14,"./js/bit2c.js":15,"./js/bitbay.js":16,"./js/bitcoincoid.js":17,"./js/bitfinex.js":18,"./js/bitfinex2.js":19,"./js/bitflyer.js":20,"./js/bithumb.js":21,"./js/bitlish.js":22,"./js/bitmarket.js":23,"./js/bitmex.js":24,"./js/bitso.js":25,"./js/bitstamp.js":26,"./js/bitstamp1.js":27,"./js/bittrex.js":28,"./js/bl3p.js":29,"./js/bleutrade.js":30,"./js/braziliex.js":31,"./js/btcbox.js":32,"./js/btcchina.js":33,"./js/btcexchange.js":34,"./js/btcmarkets.js":35,"./js/btctradeua.js":36,"./js/btcturk.js":37,"./js/btcx.js":38,"./js/bter.js":39,"./js/bxinth.js":40,"./js/ccex.js":41,"./js/cex.js":42,"./js/chbtc.js":43,"./js/chilebit.js":44,"./js/coincheck.js":45,"./js/coinexchange.js":46,"./js/coinfloor.js":47,"./js/coingi.js":48,"./js/coinmarketcap.js":49,"./js/coinmate.js":50,"./js/coinsecure.js":51,"./js/coinspot.js":52,"./js/cryptopia.js":53,"./js/dsx.js":54,"./js/exmo.js":55,"./js/flowbtc.js":56,"./js/foxbit.js":57,"./js/fybse.js":58,"./js/fybsg.js":59,"./js/gatecoin.js":60,"./js/gateio.js":61,"./js/gdax.js":62,"./js/gemini.js":63,"./js/getbtc.js":64,"./js/hitbtc.js":65,"./js/hitbtc2.js":66,"./js/huobi.js":67,"./js/huobicny.js":68,"./js/huobipro.js":69,"./js/independentreserve.js":70,"./js/itbit.js":71,"./js/jubi.js":72,"./js/kraken.js":73,"./js/kucoin.js":74,"./js/kuna.js":75,"./js/lakebtc.js":76,"./js/liqui.js":77,"./js/livecoin.js":78,"./js/luno.js":79,"./js/lykke.js":80,"./js/mercado.js":81,"./js/mixcoins.js":82,"./js/nova.js":83,"./js/okcoincny.js":84,"./js/okcoinusd.js":85,"./js/okex.js":86,"./js/paymium.js":87,"./js/poloniex.js":88,"./js/qryptos.js":89,"./js/quadrigacx.js":90,"./js/quoinex.js":91,"./js/southxchange.js":92,"./js/surbitcoin.js":93,"./js/therock.js":94,"./js/tidex.js":95,"./js/urdubit.js":96,"./js/vaultoro.js":97,"./js/vbtc.js":98,"./js/virwox.js":99,"./js/wex.js":100,"./js/xbtce.js":101,"./js/yobit.js":102,"./js/yunbi.js":103,"./js/zaif.js":104,"./js/zb.js":105}],3:[function(require,module,exports){ +},{"./js/_1broker.js":3,"./js/_1btcxe.js":4,"./js/acx.js":5,"./js/allcoin.js":6,"./js/anxpro.js":7,"./js/base/Exchange":8,"./js/base/errors":10,"./js/base/functions":11,"./js/bibox.js":22,"./js/binance.js":23,"./js/bit2c.js":24,"./js/bitbay.js":25,"./js/bitcoincoid.js":26,"./js/bitfinex.js":27,"./js/bitfinex2.js":28,"./js/bitflyer.js":29,"./js/bithumb.js":30,"./js/bitlish.js":31,"./js/bitmarket.js":32,"./js/bitmex.js":33,"./js/bitso.js":34,"./js/bitstamp.js":35,"./js/bitstamp1.js":36,"./js/bittrex.js":37,"./js/bl3p.js":38,"./js/bleutrade.js":39,"./js/braziliex.js":40,"./js/btcbox.js":41,"./js/btcchina.js":42,"./js/btcexchange.js":43,"./js/btcmarkets.js":44,"./js/btctradeua.js":45,"./js/btcturk.js":46,"./js/btcx.js":47,"./js/bter.js":48,"./js/bxinth.js":49,"./js/ccex.js":50,"./js/cex.js":51,"./js/chbtc.js":52,"./js/chilebit.js":53,"./js/coincheck.js":54,"./js/coinexchange.js":55,"./js/coinfloor.js":56,"./js/coingi.js":57,"./js/coinmarketcap.js":58,"./js/coinmate.js":59,"./js/coinsecure.js":60,"./js/coinspot.js":61,"./js/cryptopia.js":62,"./js/dsx.js":63,"./js/exmo.js":64,"./js/flowbtc.js":65,"./js/foxbit.js":66,"./js/fybse.js":67,"./js/fybsg.js":68,"./js/gatecoin.js":69,"./js/gateio.js":70,"./js/gdax.js":71,"./js/gemini.js":72,"./js/getbtc.js":73,"./js/hitbtc.js":74,"./js/hitbtc2.js":75,"./js/huobi.js":76,"./js/huobicny.js":77,"./js/huobipro.js":78,"./js/independentreserve.js":79,"./js/itbit.js":80,"./js/jubi.js":81,"./js/kraken.js":82,"./js/kucoin.js":83,"./js/kuna.js":84,"./js/lakebtc.js":85,"./js/liqui.js":86,"./js/livecoin.js":87,"./js/luno.js":88,"./js/lykke.js":89,"./js/mercado.js":90,"./js/mixcoins.js":91,"./js/nova.js":92,"./js/okcoincny.js":93,"./js/okcoinusd.js":94,"./js/okex.js":95,"./js/paymium.js":96,"./js/poloniex.js":97,"./js/qryptos.js":98,"./js/quadrigacx.js":99,"./js/quoinex.js":100,"./js/southxchange.js":101,"./js/surbitcoin.js":102,"./js/therock.js":103,"./js/tidex.js":104,"./js/urdubit.js":105,"./js/vaultoro.js":106,"./js/vbtc.js":107,"./js/virwox.js":108,"./js/wex.js":109,"./js/xbtce.js":110,"./js/yobit.js":111,"./js/yunbi.js":112,"./js/zaif.js":113,"./js/zb.js":114}],3:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -177,10 +177,12 @@ module.exports = class _1broker extends Exchange { 'countries': 'US', 'rateLimit': 1500, 'version': 'v2', - 'hasPublicAPI': false, - 'hasCORS': true, - 'hasFetchTrades': false, - 'hasFetchOHLCV': true, + 'has': { + 'publicAPI': false, + 'CORS': true, + 'fetchTrades': false, + 'fetchOHLCV': true, + }, 'timeframes': { '1m': '60', '15m': '900', @@ -441,9 +443,11 @@ module.exports = class _1btcxe extends Exchange { 'name': '1BTCXE', 'countries': 'PA', // Panama 'comment': 'Crypto Capital API', - 'hasCORS': true, - 'hasFetchOHLCV': true, - 'hasWithdraw': true, + 'has': { + 'CORS': true, + 'fetchTickers': true, + 'withdraw': true, + }, 'timeframes': { '1d': '1year', }, @@ -697,10 +701,12 @@ module.exports = class acx extends Exchange { 'countries': 'AU', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': true, - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, - 'hasWithdraw': true, + 'has': { + 'CORS': true, + 'fetchTickers': true, + 'fetchOHLCV': true, + 'withdraw': true, + }, 'timeframes': { '1m': '1', '5m': '5', @@ -1094,7 +1100,9 @@ module.exports = class allcoin extends okcoinusd { 'id': 'allcoin', 'name': 'Allcoin', 'countries': 'CA', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'extension': '', 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/31561809-c316b37c-b061-11e7-8d5a-b547b4d730eb.jpg', @@ -1190,7 +1198,7 @@ module.exports = class allcoin extends okcoinusd { } } -},{"./okcoinusd.js":85}],7:[function(require,module,exports){ +},{"./okcoinusd.js":94}],7:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -1209,9 +1217,11 @@ module.exports = class anxpro extends Exchange { 'countries': [ 'JP', 'SG', 'HK', 'NZ' ], 'version': '2', 'rateLimit': 1500, - 'hasCORS': false, - 'hasFetchTrades': false, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'fetchTrades': false, + 'withdraw': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27765983-fd8595da-5ec9-11e7-82e3-adb3ab8c2612.jpg', 'api': 'https://anxpro.com/api', @@ -1423,25 +1433,31 @@ module.exports = class anxpro extends Exchange { (function (process){ "use strict"; -//----------------------------------------------------------------------------- +/* ------------------------------------------------------------------------ */ -const isNode = (typeof window === 'undefined') && !(typeof WorkerGlobalScope !== 'undefined' && self instanceof WorkerGlobalScope) - , functions = require ('./functions') - , throttle = require ('./throttle') - , defaultFetch = isNode ? require ('fetch-ponyfill')().fetch : fetch +const functions = require ('./functions') , Market = require ('./Market') -const { deepExtend +const { isNode + , keys + , values + , deepExtend , extend - , sleep - , timeout , flatten , indexBy , sortBy , groupBy , aggregate , uuid - , precisionFromString } = functions + , unCamelCase + , precisionFromString + , throttle + , capitalize } = functions + +const { now + , sleep + , timeout + , TimedOut } = require ('./functions/time') const { ExchangeError , NotSupported @@ -1450,9 +1466,11 @@ const { ExchangeError , RequestTimeout , ExchangeNotAvailable } = require ('./errors') -// stub until we get a better solution for Webpack and React -// const journal = isNode && require ('./journal') -const journal = undefined +const defaultFetch = isNode ? require ('fetch-ponyfill')().fetch : fetch + +const journal = undefined // isNode && require ('./journal') // stub until we get a better solution for Webpack and React + +/* ------------------------------------------------------------------------ */ module.exports = class Exchange { @@ -1503,9 +1521,9 @@ module.exports = class Exchange { this.iso8601 = timestamp => new Date (timestamp).toISOString () this.parse8601 = x => Date.parse (((x.indexOf ('+') >= 0) || (x.slice (-1) == 'Z')) ? x : (x + 'Z')) - this.milliseconds = Date.now - this.microseconds = () => Math.floor (this.milliseconds () * 1000) - this.seconds = () => Math.floor (this.milliseconds () / 1000) + this.milliseconds = now + this.microseconds = () => now () * 1000 // TODO: utilize performance.now for that purpose + this.seconds = () => Math.floor (now () / 1000) this.id = undefined // rate limiter settings @@ -1526,26 +1544,6 @@ module.exports = class Exchange { this.userAgent = undefined this.twofa = false // two-factor authentication (2FA) this.timeframes = undefined - this.hasPublicAPI = true - this.hasPrivateAPI = true - this.hasCORS = false - this.hasDeposit = false - this.hasFetchBalance = true - this.hasFetchClosedOrders = false - this.hasFetchCurrencies = false - this.hasFetchMyTrades = false - this.hasFetchOHLCV = false - this.hasFetchOpenOrders = false - this.hasFetchOrder = false - this.hasFetchOrderBook = true - this.hasFetchOrders = false - this.hasFetchTicker = true - this.hasFetchTickers = false - this.hasFetchBidsAsks = false - this.hasFetchTrades = true - this.hasWithdraw = false - this.hasCreateOrder = this.hasPrivateAPI - this.hasCancelOrder = this.hasPrivateAPI this.apiKey = undefined this.secret = undefined @@ -1575,68 +1573,23 @@ module.exports = class Exchange { this.arrayConcat = (a, b) => a.concat (b) - // TODO: generate - - this.market_id = this.marketId - this.market_ids = this.marketIds - this.array_concat = this.arrayConcat - this.implode_params = this.implodeParams - this.extract_params = this.extractParams - this.fetch_balance = this.fetchBalance - this.fetch_free_balance = this.fetchFreeBalance - this.fetch_used_balance = this.fetchUsedBalance - this.fetch_total_balance = this.fetchTotalBalance - this.fetch_l2_order_book = this.fetchL2OrderBook - this.fetch_order_book = this.fetchOrderBook - this.fetch_bids_asks = this.fetchBidsAsks - this.fetch_tickers = this.fetchTickers - this.fetch_ticker = this.fetchTicker - this.fetch_trades = this.fetchTrades - this.fetch_order = this.fetchOrder - this.fetch_orders = this.fetchOrders - this.fetch_open_orders = this.fetchOpenOrders - this.fetch_closed_orders = this.fetchClosedOrders - this.fetch_order_status = this.fetchOrderStatus - this.fetch_markets = this.fetchMarkets - this.load_markets = this.loadMarkets - this.set_markets = this.setMarkets - this.parse_balance = this.parseBalance - this.parse_bid_ask = this.parseBidAsk - this.parse_bids_asks = this.parseBidsAsks - this.parse_order_book = this.parseOrderBook - this.parse_trades = this.parseTrades - this.parse_orders = this.parseOrders - this.parse_ohlcv = this.parseOHLCV - this.parse_ohlcvs = this.parseOHLCVs - this.edit_limit_buy_order = this.editLimitBuyOrder - this.edit_limit_sell_order = this.editLimitSellOrder - this.edit_limit_order = this.editLimitOrder - this.edit_order = this.editOrder - this.create_limit_buy_order = this.createLimitBuyOrder - this.create_limit_sell_order = this.createLimitSellOrder - this.create_market_buy_order = this.createMarketBuyOrder - this.create_market_sell_order = this.createMarketSellOrder - this.create_order = this.createOrder - this.calculate_fee = this.calculateFee - this.common_currency_code = this.commonCurrencyCode - this.price_to_precision = this.priceToPrecision - this.amount_to_precision = this.amountToPrecision - this.amount_to_string = this.amountToString - this.fee_to_precision = this.feeToPrecision - this.cost_to_precision = this.costToPrecision - this.precisionFromString = precisionFromString - this.precision_from_string = precisionFromString - this.truncate = functions.truncate - this.truncate_to_string = functions.truncate_to_string - this.uuid = uuid - - // API methods metainfo + const names = Object.getOwnPropertyNames (this).concat ( + Object.getOwnPropertyNames (this.constructor.prototype)) + + for (const k of names) + this[unCamelCase (k)] = this[k] + + /* exchange's capabilities (overrideable) */ + this.has = { - 'cancelOrder': this.hasPrivateAPI, + 'CORS': false, + 'publicAPI': true, + 'privateAPI': true, + 'cancelOrder': true, 'createDepositAddress': false, - 'createOrder': this.hasPrivateAPI, + 'createOrder': true, 'deposit': false, - 'fetchBalance': this.hasPrivateAPI, + 'fetchBalance': true, 'fetchClosedOrders': false, 'fetchCurrencies': false, 'fetchDepositAddress': false, @@ -1661,6 +1614,11 @@ module.exports = class Exchange { for (const [property, value] of Object.entries (config)) this[property] = deepExtend (this[property], value) + // generate old metainfo interface + for (const k in this.has) { + this['has' + capitalize (k)] = !!this.has[k] // converts 'emulated' to true + } + if (this.api) this.defineRestApi (this.api, 'request') @@ -1720,8 +1678,8 @@ module.exports = class Exchange { .then (response => this.handleRestResponse (response, url, method, headers, body)) return timeout (this.timeout, promise).catch (e => { - if (e instanceof RequestTimeout) - throw new RequestTimeout (this.id + ' ' + method + ' ' + url + ' ' + e.message + ' (' + this.timeout + ' ms)') + if (e instanceof TimedOut) + throw new RequestTimeout (this.id + ' ' + method + ' ' + url + ' request timed out (' + this.timeout + ' ms)') throw e }) } @@ -2324,7 +2282,7 @@ module.exports = class Exchange { }).call(this,require('_process')) -},{"./Market":9,"./errors":10,"./functions":11,"./throttle":12,"_process":141,"fetch-ponyfill":140}],9:[function(require,module,exports){ +},{"./Market":9,"./errors":10,"./functions":11,"./functions/time":20,"_process":150,"fetch-ponyfill":149}],9:[function(require,module,exports){ "use strict"; module.exports = class Market { @@ -2349,362 +2307,385 @@ module.exports = class Market { } },{}],10:[function(require,module,exports){ -class BaseError extends Error { - constructor (message) { - super (message) - // a workaround to make `instanceof BaseError` work in ES5 - this.constructor = BaseError - this.__proto__ = BaseError.prototype - this.message = message - } -} +/* ------------------------------------------------------------------------ */ -class ExchangeError extends BaseError { - constructor (message) { - super (message) - this.constructor = ExchangeError - this.__proto__ = ExchangeError.prototype - this.message = message - } -} +module.exports = subclass ( -class NotSupported extends ExchangeError { - constructor (message) { - super (message) - this.constructor = NotSupported - this.__proto__ = NotSupported.prototype - this.message = message - } -} +/* Root class */ -class AuthenticationError extends ExchangeError { - constructor (message) { - super (message) - this.constructor = AuthenticationError - this.__proto__ = AuthenticationError.prototype - this.message = message - } -} + Error, -class InvalidNonce extends ExchangeError { - constructor (message) { - super (message) - this.constructor = InvalidNonce - this.__proto__ = InvalidNonce.prototype - this.message = message - } -} +/* Derived class hierarchy */ -class InsufficientFunds extends ExchangeError { - constructor (message) { - super (message) - this.constructor = InsufficientFunds - this.__proto__ = InsufficientFunds.prototype - this.message = message + { 'BaseError': + { 'ExchangeError': + { 'NotSupported': {} + , 'AuthenticationError': {} + , 'InvalidNonce': {} + , 'InsufficientFunds': {} + , 'InvalidOrder': + { 'OrderNotFound': {} + , 'OrderNotCached': {} + , 'CancelPending': {} + } + , 'NetworkError': + { 'DDoSProtection': {} + , 'RequestTimeout': {} + , 'ExchangeNotAvailable': {} + } + } + } } -} +) -class InvalidOrder extends ExchangeError { - constructor (message) { - super (message) - this.constructor = InvalidOrder - this.__proto__ = InvalidOrder.prototype - this.message = message - } -} +/* ------------------------------------------------------------------------ */ -class OrderNotFound extends InvalidOrder { - constructor (message) { - super (message) - this.constructor = OrderNotFound - this.__proto__ = OrderNotFound.prototype - this.message = message - } -} +function subclass (BaseClass, classes, namespace = {}) { -class OrderNotCached extends InvalidOrder { - constructor (message) { - super (message) - this.constructor = OrderNotCached - this.__proto__ = OrderNotCached.prototype - this.message = message - } -} + for (const [$class, subclasses] of Object.entries (classes)) { -class CancelPending extends InvalidOrder { - constructor (message) { - super (message) - this.constructor = CancelPending - this.__proto__ = CancelPending.prototype - this.message = message - } -} + const Class = Object.assign (namespace, { + + /* By creating a named property, we trick compiler to assign our class constructor function a name. + Otherwise, all our error constructors would be shown as [Function: Error] in the debugger! And + the super-useful `e.constructor.name` magic wouldn't work — we then would have no chance to + obtain a error type string from an error instance programmatically! */ -class NetworkError extends BaseError { - constructor (message) { - super (message) - this.constructor = NetworkError - this.__proto__ = NetworkError.prototype - this.message = message + [$class]: class extends BaseClass { + + constructor (message) { + + super (message) + + /* A workaround to make `instanceof` work on custom Error classes in transpiled ES5. + See my blog post for the explanation of this hack: + + https://medium.com/@xpl/javascript-deriving-from-error-properly-8d2f8f315801 */ + + this.constructor = Class + this.__proto__ = Class.prototype + this.message = message + } + } + + })[$class] + + subclass (Class, subclasses, namespace) } + + return namespace } -class DDoSProtection extends NetworkError { - constructor (message) { - super (message) - this.constructor = DDoSProtection - this.__proto__ = DDoSProtection.prototype - this.message = message - } +/* ------------------------------------------------------------------------ */ + +},{}],11:[function(require,module,exports){ +"use strict"; + +/* ------------------------------------------------------------------------ */ + +const { unCamelCase } = require ('./functions/string') + +const unCamelCasePropertyNames = x => { + for (const k in x) x[unCamelCase (k)] = x[k] // camel_case_method = camelCaseMethod + return x + } + +/* ------------------------------------------------------------------------ */ + +module.exports = unCamelCasePropertyNames (Object.assign ({} + + , require ('./functions/platform') + , require ('./functions/generic') + , require ('./functions/string') + , require ('./functions/type') + , require ('./functions/number') + , require ('./functions/encode') + , require ('./functions/crypto') + , require ('./functions/time') + , require ('./functions/throttle') + , require ('./functions/misc') +)) + +/* ------------------------------------------------------------------------ */ + +},{"./functions/crypto":12,"./functions/encode":13,"./functions/generic":14,"./functions/misc":15,"./functions/number":16,"./functions/platform":17,"./functions/string":18,"./functions/throttle":19,"./functions/time":20,"./functions/type":21}],12:[function(require,module,exports){ +"use strict"; + +/* ------------------------------------------------------------------------ */ + +const CryptoJS = require ('crypto-js') +const { capitalize } = require ('./string') +const { stringToBase64, utf16ToBase64, urlencodeBase64 } = require ('./encode') + +/* ------------------------------------------------------------------------ */ + +const hash = (request, hash = 'md5', digest = 'hex') => { + const result = CryptoJS[hash.toUpperCase ()] (request) + return (digest == 'binary') ? result : result.toString (CryptoJS.enc[capitalize (digest)]) } -class RequestTimeout extends NetworkError { - constructor (message) { - super (message) - this.constructor = RequestTimeout - this.__proto__ = RequestTimeout.prototype - this.message = message - } +/* ............................................. */ + +const hmac = (request, secret, hash = 'sha256', digest = 'hex') => { + const encoding = (digest == 'binary') ? 'Latin1' : capitalize (digest) + return CryptoJS['Hmac' + hash.toUpperCase ()] (request, secret).toString (CryptoJS.enc[capitalize (encoding)]) } -class ExchangeNotAvailable extends NetworkError { - constructor (message) { - super (message) - this.constructor = ExchangeNotAvailable - this.__proto__ = ExchangeNotAvailable.prototype - this.message = message - } +/* ............................................. */ + +const jwt = function JSON_web_token (request, secret, alg = 'HS256', hash = 'sha256') { + const encodedHeader = urlencodeBase64 (stringToBase64 (JSON.stringify ({ 'alg': alg, 'typ': 'JWT' }))) + , encodedData = urlencodeBase64 (stringToBase64 (JSON.stringify (request))) + , token = [ encodedHeader, encodedData ].join ('.') + , signature = urlencodeBase64 (utf16ToBase64 (hmac (token, secret, hash, 'utf16'))) + return [ token, signature ].join ('.') } +/* ------------------------------------------------------------------------ */ + module.exports = { - BaseError, - ExchangeError, - NotSupported, - AuthenticationError, - InvalidNonce, - InsufficientFunds, - InvalidOrder, - OrderNotFound, - OrderNotCached, - CancelPending, - NetworkError, - DDoSProtection, - RequestTimeout, - ExchangeNotAvailable, + hash, + hmac, + jwt } -},{}],11:[function(require,module,exports){ +/* ------------------------------------------------------------------------ */ + +},{"./encode":13,"./string":18,"crypto-js":123}],13:[function(require,module,exports){ "use strict"; -//----------------------------------------------------------------------------- +/* ------------------------------------------------------------------------ */ const CryptoJS = require ('crypto-js') - , qs = require ('qs') // querystring - -//----------------------------------------------------------------------------- +const qs = require ('qs') // querystring (TODO: get rid of that dependency) -const { RequestTimeout } = require ('./errors') +/* ------------------------------------------------------------------------ */ -//----------------------------------------------------------------------------- -// utility helpers +module.exports = + + { json: JSON.stringify + , unjson: JSON.parse -const setTimeout_original = setTimeout + , stringToBinary (str) { + const arr = new Uint8Array (str.length) + for (let i = 0; i < str.length; i++) { arr[i] = str.charCodeAt (i); } + return CryptoJS.lib.WordArray.create (arr) + } -// setTimeout can fire earlier than specified, so we need to ensure it does not happen... + , stringToBase64: string => CryptoJS.enc.Latin1.parse (string).toString (CryptoJS.enc.Base64) + , utf16ToBase64: string => CryptoJS.enc.Utf16 .parse (string).toString (CryptoJS.enc.Base64) + , base64ToBinary: string => CryptoJS.enc.Base64.parse (string) + , base64ToString: string => CryptoJS.enc.Base64.parse (string).toString (CryptoJS.enc.Utf8) + , binaryToString: string => string -const setTimeout_safe = (done, ms, setTimeout = setTimeout_original /* overrideable for mocking purposes */, targetTime = Date.now () + ms) => { + , binaryConcat: (...args) => args.reduce ((a, b) => a.concat (b)) - let clearInnerTimeout = () => {} - let active = true + , urlencode: object => qs.stringify (object) + , rawencode: object => qs.stringify (object, { encode: false }) - let id = setTimeout (() => { - active = true - const rest = targetTime - Date.now () - if (rest > 0) { - clearInnerTimeout = setTimeout_safe (done, rest, setTimeout, targetTime) // try sleep more - } else { - done () - } - }, ms) + // Url-safe-base64 without equals signs, with + replaced by - and slashes replaced by underscores - return function clear () { - if (active) { - active = false // dunno if IDs are unique on various platforms, so it's better to rely on this flag to exclude the possible cancellation of the wrong timer (if called after completion) - clearTimeout (id) - } - clearInnerTimeout () - } + , urlencodeBase64: base64string => base64string.replace (/[=]+$/, '') + .replace (/\+/g, '-') + .replace (/\//g, '_') } -const sleep = ms => new Promise (resolve => setTimeout_safe (resolve, ms)) +/* ------------------------------------------------------------------------ */ -const decimal = float => parseFloat (float).toString () +},{"crypto-js":123,"qs":152}],14:[function(require,module,exports){ +"use strict"; -const timeout = async (ms, promise) => { +/* ------------------------------------------------------------------------ */ - let clear = () => {} - const timeout = new Promise (resolve => (clear = setTimeout_safe (resolve, ms))) +const { isObject, isNumber, isDictionary, isArray } = require ('./type') - try { - return await Promise.race ([promise, timeout.then (() => { throw new RequestTimeout ('request timed out') })]) - } finally { - clear () // fixes https://github.com/ccxt/ccxt/issues/749 - } -} +/* ------------------------------------------------------------------------ */ -const capitalize = string => string.length ? (string.charAt (0).toUpperCase () + string.slice (1)) : string +const empty = () => Object.create (null) // empty obj without even a prototype -const keysort = object => { - const result = {} - Object.keys (object).sort ().forEach (key => result[key] = object[key]) - return result -} + , keys = Object.keys -const extend = (...args) => Object.assign ({}, ...args) + , values = x => !isArray (x) // don't copy arrays if they're already arrays! + ? Object.values (x) + : x -const deepExtend = function (...args) { + , index = x => new Set (values (x)) - // if (args.length < 1) - // return args - // else if (args.length < 2) - // return args[0] + , extend = (...args) => Object.assign (empty (), ...args) // NB: side-effect free - let result = undefined + , clone = x => isArray (x) + ? Array.from (x) // clones arrays + : extend (x) // clones objects - for (const arg of args) { +/* ------------------------------------------------------------------------ */ - if (arg && (typeof arg === 'object') && (arg.constructor === Object || !('constructor' in arg))) { +module.exports = { - if (typeof result !== 'object') { - result = {} - } + empty + , keys + , values + , extend + , clone + , index + , ordered: x => x // a stub to keep assoc keys in order (in JS it does nothing, it's mostly for Python) + , unique: x => Array.from (index (x)) + +/* ............................................. */ + + , keysort (x, out = empty ()) { + + for (const k of keys (x).sort ()) + out[k] = x[k] + + return out + } - for (const key in arg) { - result[key] = deepExtend (result[key], arg[key]) - } +/* ............................................. */ - } else { + , indexBy (x, k, out = empty ()) { - result = arg - } + for (const v of values (x)) + if (k in v) + out[v[k]] = v + + return out } - return result -} +/* ............................................. */ -const omit = (object, ...args) => { - const result = extend (object) - for (const x of args) { - if (typeof x === 'string') { - delete result[x] - } else if (Array.isArray (x)) { - for (const k of x) - delete result[k] + , groupBy (x, k, out = empty ()) { + + for (const v of values (x)) { + if (k in v) { + const p = v[k] + out[p] = out[p] || [] + out[p].push (v) + } } + return out } - return result -} -const groupBy = (array, key) => { - const result = {} - Object - .values (array) - .filter (entry => typeof entry[key] != 'undefined') - .forEach (entry => { - if (typeof result[entry[key]] == 'undefined') - result[entry[key]] = [] - result[entry[key]].push (entry) - }) - return result -} +/* ............................................. */ + + , filterBy (x, k, value = undefined, out = []) { -const filterBy = (array, key, value = undefined) => { - if (value) { - let grouped = groupBy (array, key) - if (value in grouped) - return grouped[value] - return [] + for (const v of values (x)) + if (v[k] === value) + out.push (v) + + return out } - return array -} -const indexBy = (array, key) => { - const result = {} - Object - .values (array) - .filter (entry => typeof entry[key] != 'undefined') - .forEach (entry => { - result[entry[key]] = entry - }) - return result -} +/* ............................................. */ -const sortBy = (array, key, descending = false) => { - descending = descending ? -1 : 1 - return array.sort ((a, b) => ((a[key] < b[key]) ? -descending : ((a[key] > b[key]) ? descending : 0))) -} + , sortBy: (array, // NB: MUTATES ARRAY! + key, + descending = false, + direction = descending ? -1 : 1) => array.sort ((a, b) => + ((a[key] < b[key]) ? -direction : + ((a[key] > b[key]) ? direction : 0))) -const flatten = (array, result = []) => { - for (let i = 0, length = array.length; i < length; i++) { - const value = array[i] - if (Array.isArray (value)) { - flatten (value, result) - } else { - result.push (value) +/* ............................................. */ + + , flatten: function flatten (x, out = []) { + + for (const v of x) { + if (isArray (v)) flatten (v, out) + else out.push (v) } + + return out } - return result -} -const unique = array => array.filter ((value, index, self) => (self.indexOf (value) == index)) +/* ............................................. */ -const pluck = (array, key) => array - .filter (element => (typeof element[key] != 'undefined')) - .map (element => element[key]) + , pluck: (x, k) => values (x) + .filter (v => k in v) + .map (v => v[k]) -const urlencode = object => qs.stringify (object) -const rawencode = object => qs.stringify (object, { encode: false }) +/* ............................................. */ -const sum = (...args) => { - const result = args.filter (arg => typeof arg !== 'undefined') - return (result.length > 0) ? - result.reduce ((sum, value) => sum + value, 0) : undefined -} + , omit (x, ...args) { + + const out = clone (x) + + for (const k of args) { + + if (isArray (k)) // omit (x, ['a', 'b']) + for (const kk of k) + delete out[kk] -const safeFloat = (object, key, defaultValue = undefined) => { - if (key in object) { - if (typeof object[key] == 'number') - return object[key] - else if ((typeof object[key] == 'string') && object[key]) - return parseFloat (object[key]) + else delete out[k] // omit (x, 'a', 'b') + } + + return out } - return defaultValue -} -const safeString = (object, key, defaultValue = undefined) => { - if (!object || !(key in object)) - return defaultValue; - let stringVal = object[key]; - if (!stringVal && typeof stringVal != 'string' && typeof stringVal != 'number') - return defaultValue; - return stringVal.toString (); -} +/* ............................................. */ + + , sum (...xs) { + + const ns = xs.filter (isNumber) // leave only numbers + + return (ns.length > 0) + ? ns.reduce ((a, b) => a + b, 0) + : undefined + } + +/* ............................................. */ + + , deepExtend: function deepExtend (...xs) { + + let out = undefined + + for (const x of xs) { + + if (isDictionary (x)) { + + if (!isObject (out)) + out = empty () + + for (const k in x) + out[k] = deepExtend (out[k], x[k]) + + } else out = x + } + + return out + } + +/* ------------------------------------------------------------------------ */ -const safeInteger = (object, key, defaultValue = undefined) => { - if (!object || !(key in object)) - return defaultValue; - let intVal = parseInt (object[key], 10); - return isNaN (intVal) ? defaultValue : intVal; } -const safeValue = (object, key, defaultValue = undefined) => { - return (object && (key in object) && object[key]) ? object[key] : defaultValue +},{"./type":21}],15:[function(require,module,exports){ +"use strict"; + +/* ------------------------------------------------------------------------ */ + +module.exports = { + + aggregate (bidasks) { + + let result = {} + + for (const [price, volume] of bidasks) { + if (volume > 0) + result[price] = (result[price] || 0) + volume + } + + return Object.keys (result) + .map (price => [parseFloat (price), + parseFloat (result[price])]) + } } -const uuid = a => a ? - (a ^ Math.random () * 16 >> a / 4).toString (16) : - ([1e7]+-1e3+-4e3+-8e3+-1e11).replace (/[018]/g, uuid) +/* ------------------------------------------------------------------------ */ + +},{}],16:[function(require,module,exports){ +/* NB: A LEGACY CODE, WILL BE RE-WRITTEN VERY SOON + ------------------------------------------------------------------------ */ // See https://stackoverflow.com/questions/1685680/how-to-avoid-scientific-notation-for-large-numbers-in-javascript for discussion @@ -2749,198 +2730,379 @@ const precisionFromString = (string) => { return (split.length > 1) ? (split[1].length) : 0 } -const ordered = x => x // a stub to keep assoc keys in order, in JS it does nothing, it's mostly for Python +module.exports = { -const aggregate = function (bidasks) { + toFixed, + truncate_to_string, + truncate, + precisionFromString +} - let result = {} +// "use strict"; - bidasks.forEach (([ price, volume ]) => { - if (volume > 0) - result[price] = (result[price] || 0) + volume - }) +// const { isString, isNumber } = require ('./type') - return Object.keys (result).map (price => [ - parseFloat (price), - parseFloat (result[price]), - ]) -} +// /* ------------------------------------------------------------------------ */ -//----------------------------------------------------------------------------- -// string ←→ binary ←→ base64 conversion routines +// const decimal = float => parseFloat (float).toString () -const stringToBinary = str => { - const arr = new Uint8Array (str.length) - for (let i = 0; i < str.length; i++) { arr[i] = str.charCodeAt(i); } - return CryptoJS.lib.WordArray.create (arr) -} +// /* ------------------------------------------------------------------------ */ -const stringToBase64 = string => CryptoJS.enc.Latin1.parse (string).toString (CryptoJS.enc.Base64) - , utf16ToBase64 = string => CryptoJS.enc.Utf16 .parse (string).toString (CryptoJS.enc.Base64) - , base64ToBinary = string => CryptoJS.enc.Base64.parse (string) - , base64ToString = string => CryptoJS.enc.Base64.parse (string).toString (CryptoJS.enc.Utf8) - , binaryToString = string => string +// // See https://stackoverflow.com/questions/1685680/how-to-avoid-scientific-notation-for-large-numbers-in-javascript for discussion -const binaryConcat = (...args) => args.reduce ((a, b) => a.concat (b)) +// function numberToString (x) { // avoid scientific notation for too large and too small numbers -// url-safe-base64 without equals signs, with + replaced by - and slashes replaced by underscores -const urlencodeBase64 = base64string => base64string.replace (/[=]+$/, '') - .replace (/\+/g, '-') - .replace (/\//g, '_') +// if (isString (x)) return x -//----------------------------------------------------------------------------- -// cryptography +// if (Math.abs (x) < 1.0) { +// const e = parseInt (x.toString ().split ('e-')[1]) +// if (e) { +// x *= Math.pow (10, e-1) +// x = '0.' + (new Array (e)).join ('0') + x.toString ().substring (2) +// } +// } else { +// let e = parseInt (x.toString ().split ('+')[1]) +// if (e > 20) { +// e -= 20 +// x /= Math.pow (10, e) +// x += (new Array (e+1)).join ('0') +// } +// } +// return x.toString () +// } -const hash = (request, hash = 'md5', digest = 'hex') => { - const result = CryptoJS[hash.toUpperCase ()] (request) - return (digest == 'binary') ? result : result.toString (CryptoJS.enc[capitalize (digest)]) -} +// /* ------------------------------------------------------------------------ */ -const hmac = (request, secret, hash = 'sha256', digest = 'hex') => { - const encoding = (digest == 'binary') ? 'Latin1' : capitalize (digest) - return CryptoJS['Hmac' + hash.toUpperCase ()] (request, secret).toString (CryptoJS.enc[capitalize (encoding)]) -} +// const padWithZeroes = (x, digits = 0) => { -//----------------------------------------------------------------------------- -// a JSON Web Token authentication method +// const [int, frac = ''] = x.split ('.') -const jwt = (request, secret, alg = 'HS256', hash = 'sha256') => { - const encodedHeader = urlencodeBase64 (stringToBase64 (JSON.stringify ({ 'alg': alg, 'typ': 'JWT' }))) - , encodedData = urlencodeBase64 (stringToBase64 (JSON.stringify (request))) - , token = [ encodedHeader, encodedData ].join ('.') - , signature = urlencodeBase64 (utf16ToBase64 (hmac (token, secret, hash, 'utf16'))) - return [ token, signature ].join ('.') -} +// return int + ((frac || (digits > 0)) +// ? ('.' + frac.padEnd (digits, '0')) +// : '') +// } -//----------------------------------------------------------------------------- +// /* ------------------------------------------------------------------------ */ -module.exports = { +// const roundDecimalString = (s, to, afterDot = false) => { - setTimeout_safe, - - // common utility functions - - sleep, - timeout, - capitalize, - keysort, - extend, - deepExtend, - omit, - groupBy, - indexBy, - sortBy, - filterBy, - flatten, - unique, - pluck, - urlencode, - rawencode, - sum, - decimal, - safeFloat, - safeString, - safeInteger, - safeValue, - ordered, - aggregate, - truncate, - truncate_to_string, - uuid, - precisionFromString, - - // underscore aliases - - index_by: indexBy, - sort_by: sortBy, - group_by: groupBy, - filter_by: filterBy, - safe_float: safeFloat, - safe_string: safeString, - safe_integer: safeInteger, - safe_value: safeValue, - - // crypto functions - - binaryConcat, - stringToBinary, - binaryToString, - stringToBase64, - utf16ToBase64, - base64ToBinary, - base64ToString, - urlencodeBase64, - hash, - hmac, - jwt, +// const digits = Array.from (s) +// const result = [] +// const dot = s.indexOf ('.') + +// let memo = 0 + +// if (afterDot) to = ((dot >= 0) ? dot : digits.length) + to + +// for (let i = digits.length - 1; i >= 0; i--) { +// const d = digits[i] +// if (d !== '.') { +// let n = (d.charCodeAt (0) - 48) + memo +// let numDigitsAhead = i +// let dotAhead = (dot >= 0) && (i >= dot) +// if ((numDigitsAhead + (dotAhead ? -1 : 0)) >= to) { // ignore dot when counting digits ahead +// n = (n > 5) ? 10 : 0 // rounding on per-digit basis +// } +// if (n > 9) { n = 0; memo = 1; } +// else memo = 0 +// digits[i] = n +// } +// } +// return (memo || '') + digits.join ('') +// } + +// /* ------------------------------------------------------------------------ */ + +// const roundNumber = (x, { digits = 8, fixed = true }) => { // accepts either strings or Numbers + +// const s = numberToString (x) + +// if (fixed) { +// return roundDecimalString (s, digits, true) + +// } else { +// const [,zeros,significantPart] = s.match (/^([^1-9]*)(.+)$/) +// return zeros + roundDecimalString (significantPart, digits) +// } +// } + +// /* ------------------------------------------------------------------------ */ + +// // See https://stackoverflow.com/questions/4912788/truncate-not-round-off-decimal-numbers-in-javascript for discussion + +// // > So, after all it turned out, rounding bugs will always haunt you, no matter how hard you try to compensate them. +// // > Hence the problem should be attacked by representing numbers exactly in decimal notation. - // json - json: JSON.stringify, - unjson: JSON.parse +// const regexCache = [] +// const truncNumber = (x, { digits = 0, fixed = true }) => { // accepts either strings or Numbers + +// const s = numberToString (x) + +// if (digits > 0) { +// const re = regexCache[digits] || (regexCache[digits] = new RegExp("([-]*\\d+\\.\\d{" + digits + "})(\\d)")) +// const [,result] = s.match (re) || [null, s] +// return fixed +// ? padWithZeroes (result, digits) +// : result + +// } else { +// throw new Error ('not implemented yet') +// } +// } + +// /* ------------------------------------------------------------------------ */ + +// const precisionFromString = (string) => { +// const split = string.replace (/0+$/g, '').split ('.') +// return (split.length > 1) ? (split[1].length) : 0 +// } + +// /* ------------------------------------------------------------------------ */ + +// const toPrecision = (x, { round = true, digits = 8, fixed = true }) => round ? roundNumber (x, { digits, fixed }) +// : truncNumber (x, { digits, fixed }) + +// /* ------------------------------------------------------------------------ */ + +// module.exports = { + +// decimal, +// numberToString, +// toPrecision, +// padWithZeroes, +// roundDecimalString, +// precisionFromString +// } + +// /* ------------------------------------------------------------------------ */ + +},{}],17:[function(require,module,exports){ +"use strict"; + +/* ------------------------------------------------------------------------ */ + +module.exports = { + + isNode: (typeof window === 'undefined') && + !((typeof WorkerGlobalScope !== 'undefined') && (self instanceof WorkerGlobalScope)) } -},{"./errors":10,"crypto-js":114,"qs":143}],12:[function(require,module,exports){ +/* ------------------------------------------------------------------------ */ + +},{}],18:[function(require,module,exports){ "use strict"; -const { sleep } = require ('./functions') +/* ------------------------------------------------------------------------ */ -const throttle = cfg => { +module.exports = + + { uuid: a => a ? (a ^ Math.random () * 16 >> a / 4).toString (16) + : ([1e7]+-1e3+-4e3+-8e3+-1e11).replace (/[018]/g, uuid) + + , unCamelCase: s => s.replace (/[a-z0-9][A-Z]/g, x => x[0] + '_' + x[1]).toLowerCase () // hasFetchOHLCV → has_fetch_ohlcv - let lastTimestamp = Date.now () - , numTokens = (typeof cfg.numTokens != 'undefined') ? cfg.numTokens : cfg.capacity - , queue = [] - , running = false - , counter = 0 + , capitalize: s => s.length + ? (s.charAt (0).toUpperCase () + s.slice (1)) + : s + } - return Object.assign (cost => { +/* ------------------------------------------------------------------------ */ - if (queue.length > cfg.maxCapacity) - throw new Error ('Backlog is over max capacity of ' + cfg.maxCapacity) +},{}],19:[function(require,module,exports){ +"use strict"; - return new Promise (async (resolve, reject) => { +/* ------------------------------------------------------------------------ */ - try { +const { sleep + , now } = require ('./time') + +/* ------------------------------------------------------------------------ */ - queue.push ({ cost, resolve, reject }) - - if (!running) { - running = true - while (queue.length > 0) { - const hasEnoughTokens = cfg.capacity ? (numTokens > 0) : (numTokens >= 0) - if (hasEnoughTokens) { - if (queue.length > 0) { - let { cost, resolve, reject } = queue[0] - cost = (cost || cfg.defaultCost) - if (numTokens >= Math.min (cost, cfg.capacity)) { - numTokens -= cost - queue.shift () - resolve () +module.exports = { + + throttle: function throttle (cfg) { + + let lastTimestamp = now () + , numTokens = (typeof cfg.numTokens != 'undefined') ? cfg.numTokens : cfg.capacity + , running = false + , counter = 0 + + const queue = [] + + return Object.assign (cost => { + + if (queue.length > cfg.maxCapacity) + throw new Error ('Backlog is over max capacity of ' + cfg.maxCapacity) + + return new Promise (async (resolve, reject) => { + + try { + queue.push ({ cost, resolve, reject }) + + if (!running) { + running = true + while (queue.length > 0) { + const hasEnoughTokens = cfg.capacity ? (numTokens > 0) : (numTokens >= 0) + if (hasEnoughTokens) { + if (queue.length > 0) { + let { cost, resolve, reject } = queue[0] + cost = (cost || cfg.defaultCost) + if (numTokens >= Math.min (cost, cfg.capacity)) { + numTokens -= cost + queue.shift () + resolve () + } } } + const t = now () + , elapsed = t - lastTimestamp + lastTimestamp = t + numTokens = Math.min (cfg.capacity, numTokens + elapsed * cfg.refillRate) + await sleep (cfg.delay) } - let now = Date.now () - let elapsed = now - lastTimestamp - lastTimestamp = now - numTokens = Math.min (cfg.capacity, numTokens + elapsed * cfg.refillRate) - await sleep (cfg.delay) + running = false } - running = false + + } catch (e) { + reject (e) } + }) - } catch (e) { + }, cfg, { configure: newCfg => throttle (Object.assign ({}, cfg, newCfg)) }) + } +} - reject (e) - } - }) +/* ------------------------------------------------------------------------ */ - }, cfg, { - configure: newCfg => throttle (Object.assign ({}, cfg, newCfg)) - }) +},{"./time":20}],20:[function(require,module,exports){ +"use strict"; + +/* ------------------------------------------------------------------------ */ + +const { isNode } = require ('./platform') + +/* ------------------------------------------------------------------------ */ + +const now = Date.now // TODO: figure out how to utilize performance.now () properly – it's not as easy as it does not return a unix timestamp... + +/* ------------------------------------------------------------------------ */ + +const setTimeout_original = setTimeout +const setTimeout_safe = (done, ms, setTimeout = setTimeout_original /* overrideable for mocking purposes */, targetTime = now () + ms) => { + +/* The built-in setTimeout function can fire its callback earlier than specified, so we + need to ensure that it does not happen: sleep recursively until `targetTime` is reached... */ + + let clearInnerTimeout = () => {} + let active = true + + let id = setTimeout (() => { + active = true + const rest = targetTime - now () + if (rest > 0) { + clearInnerTimeout = setTimeout_safe (done, rest, setTimeout, targetTime) // try sleep more + } else { + done () + } + }, ms) + + return function clear () { + if (active) { + active = false // dunno if IDs are unique on various platforms, so it's better to rely on this flag to exclude the possible cancellation of the wrong timer (if called after completion) + clearTimeout (id) + } + clearInnerTimeout () + } +} + +/* ------------------------------------------------------------------------ */ + +class TimedOut extends Error { + + constructor () { + const message = 'timed out' + super (message) + this.constructor = TimedOut + this.__proto__ = TimedOut.prototype + this.message = message + } +} + +/* ------------------------------------------------------------------------ */ + +module.exports = + + { now + , setTimeout_safe + , sleep: ms => new Promise (resolve => setTimeout_safe (resolve, ms)) + , TimedOut + , timeout: async (ms, promise) => { + + let clear = () => {} + const expires = new Promise (resolve => (clear = setTimeout_safe (resolve, ms))) + + try { + return await Promise.race ([promise, expires.then (() => { throw new TimedOut () })]) + } finally { + clear () // fixes https://github.com/ccxt/ccxt/issues/749 + } + } } -module.exports = throttle +/* ------------------------------------------------------------------------ */ + +},{"./platform":17}],21:[function(require,module,exports){ +"use strict"; + +/* ------------------------------------------------------------------------ */ + +const isNumber = Number.isFinite + , isArray = Array.isArray + , isString = s => (typeof s === 'string') + , isObject = o => (o !== null) && (typeof o === 'object') + , isDictionary = o => (isObject (o) && !isArray (o)) + , isStringCoercible = x => (hasProps (x) && x.toString) || isNumber (x) + +/* ............................................. */ + +const hasProps = o => (o !== undefined) && + (o !== null) + + , prop = (o, k) => isObject (o) ? o[k] + : undefined + +/* ............................................. */ + +const asFloat = x => (isNumber (x) || isString (x)) ? parseFloat (x) : NaN + , asInteger = x => (isNumber (x) || isString (x)) ? parseInt (x, 10) : NaN + +/* ............................................. */ + +module.exports = -},{"./functions":11}],13:[function(require,module,exports){ + { isNumber + , isArray + , isObject + , isString + , isStringCoercible + , isDictionary + + , hasProps + , prop + + , asFloat + , asInteger + + , safeFloat: (o, k, $default, n = asFloat (prop (o, k))) => isNumber (n) ? n : $default + , safeInteger: (o, k, $default, n = asInteger (prop (o, k))) => isNumber (n) ? n : $default + , safeValue: (o, k, $default, x = prop (o, k) ) => hasProps (x) ? x : $default + , safeString: (o, k, $default, x = prop (o, k) ) => isStringCoercible (x) ? String (x) : $default + + } + +/* ------------------------------------------------------------------------ */ + +},{}],22:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -2958,16 +3120,9 @@ module.exports = class bibox extends Exchange { 'name': 'Bibox', 'countries': [ 'CN', 'US', 'KR' ], 'version': 'v1', - 'hasCORS': false, - 'hasPublicAPI': false, - 'hasFetchBalance': true, - 'hasFetchCurrencies': true, - 'hasFetchTickers': true, - 'hasFetchOrders': true, - 'hasFetchMyTrades': true, - 'hasFetchOHLCV': true, - 'hasWithdraw': true, 'has': { + 'CORS': false, + 'publicAPI': false, 'fetchBalance': true, 'fetchCurrencies': true, 'fetchTickers': true, @@ -3470,7 +3625,7 @@ module.exports = class bibox extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],14:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],23:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -3488,18 +3643,9 @@ module.exports = class binance extends Exchange { 'name': 'Binance', 'countries': 'JP', // Japan 'rateLimit': 500, - 'hasCORS': false, - // obsolete metainfo interface - 'hasFetchBidsAsks': true, - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, - 'hasFetchMyTrades': true, - 'hasFetchOrder': true, - 'hasFetchOrders': true, - 'hasFetchOpenOrders': true, - 'hasWithdraw': true, // new metainfo interface 'has': { + 'CORS': false, 'fetchBidsAsks': true, 'fetchTickers': true, 'fetchOHLCV': true, @@ -4323,7 +4469,7 @@ module.exports = class binance extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],15:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],24:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -4340,7 +4486,9 @@ module.exports = class bit2c extends Exchange { 'name': 'Bit2C', 'countries': 'IL', // Israel 'rateLimit': 3000, - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766119-3593220e-5ece-11e7-8b3a-5a041f6bcc3f.jpg', 'api': 'https://www.bit2c.co.il', @@ -4517,7 +4665,7 @@ module.exports = class bit2c extends Exchange { } } -},{"./base/Exchange":8}],16:[function(require,module,exports){ +},{"./base/Exchange":8}],25:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -4535,8 +4683,10 @@ module.exports = class bitbay extends Exchange { 'name': 'BitBay', 'countries': [ 'PL', 'EU' ], // Poland 'rateLimit': 1000, - 'hasCORS': true, - 'hasWithdraw': true, + 'has': { + 'CORS': true, + 'withdraw': true + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766132-978a7bd8-5ece-11e7-9540-bc96d1e9bbb8.jpg', 'www': 'https://bitbay.net', @@ -4830,7 +4980,7 @@ module.exports = class bitbay extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],17:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],26:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -4847,19 +4997,8 @@ module.exports = class bitcoincoid extends Exchange { 'id': 'bitcoincoid', 'name': 'Bitcoin.co.id', 'countries': 'ID', // Indonesia - 'hasCORS': false, - // obsolete metainfo interface - 'hasFetchTickers': false, - 'hasFetchOHLCV': false, - 'hasFetchOrder': true, - 'hasFetchOrders': false, - 'hasFetchClosedOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchMyTrades': false, - 'hasFetchCurrencies': false, - 'hasWithdraw': false, - // new metainfo interface 'has': { + 'CORS': false, 'fetchTickers': false, 'fetchOHLCV': false, 'fetchOrder': true, @@ -5173,7 +5312,7 @@ module.exports = class bitcoincoid extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],18:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],27:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -5192,17 +5331,9 @@ module.exports = class bitfinex extends Exchange { 'countries': 'VG', 'version': 'v1', 'rateLimit': 1500, - 'hasCORS': false, - // old metainfo interface - 'hasFetchOrder': true, - 'hasFetchTickers': true, - 'hasDeposit': true, - 'hasWithdraw': true, - 'hasFetchOHLCV': true, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, // new metainfo interface 'has': { + 'CORS': false, 'fetchOHLCV': true, 'fetchTickers': true, 'fetchOrder': true, @@ -5890,7 +6021,7 @@ module.exports = class bitfinex extends Exchange { } }; -},{"./base/Exchange":8,"./base/errors":10}],19:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],28:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -5908,18 +6039,9 @@ module.exports = class bitfinex2 extends bitfinex { 'name': 'Bitfinex v2', 'countries': 'VG', 'version': 'v2', - 'hasCORS': true, - // old metainfo interface - 'hasCreateOrder': false, - 'hasFetchOrder': true, - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, - 'hasWithdraw': true, - 'hasDeposit': false, - 'hasFetchOpenOrders': false, - 'hasFetchClosedOrders': false, // new metainfo interface 'has': { + 'CORS': true, 'createOrder': false, 'fetchOHLCV': true, 'fetchTickers': true, @@ -6326,7 +6448,7 @@ module.exports = class bitfinex2 extends bitfinex { } } -},{"./base/errors":10,"./bitfinex.js":18}],20:[function(require,module,exports){ +},{"./base/errors":10,"./bitfinex.js":27}],29:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -6344,8 +6466,10 @@ module.exports = class bitflyer extends Exchange { 'countries': 'JP', 'version': 'v1', 'rateLimit': 500, - 'hasCORS': false, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'withdraw': true + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28051642-56154182-660e-11e7-9b0d-6042d1e6edd8.jpg', 'api': 'https://api.bitflyer.jp', @@ -6591,7 +6715,7 @@ module.exports = class bitflyer extends Exchange { } } -},{"./base/Exchange":8}],21:[function(require,module,exports){ +},{"./base/Exchange":8}],30:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -6609,12 +6733,8 @@ module.exports = class bithumb extends Exchange { 'name': 'Bithumb', 'countries': 'KR', // South Korea 'rateLimit': 500, - 'hasCORS': true, - // obsolete metainfo interface - 'hasFetchTickers': true, - 'hasWithdraw': true, - // new metainfo interface 'has': { + 'CORS': true, 'fetchTickers': true, 'withdraw': true, }, @@ -6939,7 +7059,7 @@ module.exports = class bithumb extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],22:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],31:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -6958,10 +7078,12 @@ module.exports = class bitlish extends Exchange { 'countries': [ 'GB', 'EU', 'RU' ], 'rateLimit': 1500, 'version': 'v1', - 'hasCORS': false, - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'fetchTickers': true, + 'fetchOHLCV': true, + 'withdraw': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766275-dcfc6c30-5ed3-11e7-839d-00a846385d0b.jpg', 'api': 'https://bitlish.com/api', @@ -7293,7 +7415,7 @@ module.exports = class bitlish extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],23:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],32:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -7311,9 +7433,11 @@ module.exports = class bitmarket extends Exchange { 'name': 'BitMarket', 'countries': [ 'PL', 'EU' ], 'rateLimit': 1500, - 'hasCORS': false, - 'hasFetchOHLCV': true, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'fetchOHLCV': true, + 'withdraw': true, + }, 'timeframes': { '90m': '90m', '6h': '6h', @@ -7666,7 +7790,7 @@ module.exports = class bitmarket extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],24:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],33:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -7686,9 +7810,11 @@ module.exports = class bitmex extends Exchange { 'version': 'v1', 'userAgent': undefined, 'rateLimit': 1500, - 'hasCORS': false, - 'hasFetchOHLCV': true, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'fetchOHLCV': true, + 'withdraw': true, + }, 'timeframes': { '1m': '1m', '5m': '5m', @@ -8100,7 +8226,7 @@ module.exports = class bitmex extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],25:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],34:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -8119,7 +8245,9 @@ module.exports = class bitso extends Exchange { 'countries': 'MX', // Mexico 'rateLimit': 2000, // 30 requests per minute 'version': 'v3', - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766335-715ce7aa-5ed5-11e7-88a8-173a27bb30fe.jpg', 'api': 'https://api.bitso.com', @@ -8365,7 +8493,7 @@ module.exports = class bitso extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],26:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],35:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -8384,12 +8512,8 @@ module.exports = class bitstamp extends Exchange { 'countries': 'GB', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': false, - // obsolete metainfo interface - 'hasFetchOrder': true, - 'hasWithdraw': true, - // new metainfo interface 'has': { + 'CORS': true, 'fetchOrder': true, 'withdraw': true, }, @@ -8807,7 +8931,7 @@ module.exports = class bitstamp extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],27:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],36:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -8826,7 +8950,9 @@ module.exports = class bitstamp1 extends Exchange { 'countries': 'GB', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27786377-8c8ab57e-5fe9-11e7-8ea4-2b05b6bcceec.jpg', 'api': 'https://www.bitstamp.net/api', @@ -9064,7 +9190,7 @@ module.exports = class bitstamp1 extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],28:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],37:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -9084,19 +9210,9 @@ module.exports = class bittrex extends Exchange { 'version': 'v1.1', 'rateLimit': 1500, 'hasAlreadyAuthenticatedSuccessfully': false, // a workaround for APIKEY_INVALID - 'hasCORS': false, - // obsolete metainfo interface - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, - 'hasFetchOrder': true, - 'hasFetchOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchMyTrades': false, - 'hasFetchCurrencies': true, - 'hasWithdraw': true, // new metainfo interface 'has': { + 'CORS': true, 'fetchTickers': true, 'fetchOHLCV': true, 'fetchOrder': true, @@ -9793,7 +9909,7 @@ module.exports = class bittrex extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],29:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],38:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -9812,7 +9928,9 @@ module.exports = class bl3p extends Exchange { 'rateLimit': 1000, 'version': '1', 'comment': 'An exchange market by BitonicNL', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28501752-60c21b82-6feb-11e7-818b-055ee6d0e754.jpg', 'api': 'https://api.bl3p.eu', @@ -9998,7 +10116,7 @@ module.exports = class bl3p extends Exchange { } } -},{"./base/Exchange":8}],30:[function(require,module,exports){ +},{"./base/Exchange":8}],39:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -10017,9 +10135,11 @@ module.exports = class bleutrade extends bittrex { 'countries': 'BR', // Brazil 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': true, - 'hasFetchTickers': true, - 'hasFetchOHLCV': false, + 'has': { + 'CORS': true, + 'fetchTickers': true, + 'fetchOHLCV': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30303000-b602dbe6-976d-11e7-956d-36c5049c01e7.jpg', 'api': { @@ -10162,7 +10282,7 @@ module.exports = class bleutrade extends bittrex { } } -},{"./base/errors":10,"./bittrex.js":28}],31:[function(require,module,exports){ +},{"./base/errors":10,"./bittrex.js":37}],40:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -10180,11 +10300,6 @@ module.exports = class braziliex extends Exchange { 'name': 'Braziliex', 'countries': 'BR', 'rateLimit': 1000, - // obsolete metainfo interface - 'hasFetchTickers': true, - 'hasFetchOpenOrders': true, - 'hasFetchMyTrades': true, - // new metainfo interface 'has': { 'fetchTickers': true, 'fetchOpenOrders': true, @@ -10631,7 +10746,7 @@ module.exports = class braziliex extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],32:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],41:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -10650,8 +10765,10 @@ module.exports = class btcbox extends Exchange { 'countries': 'JP', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': false, - 'hasFetchOHLCV': false, + 'has': { + 'CORS': false, + 'fetchOHLCV': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/31275803-4df755a8-aaa1-11e7-9abb-11ec2fad9f2d.jpg', 'api': 'https://www.btcbox.co.jp/api', @@ -10857,7 +10974,7 @@ module.exports = class btcbox extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],33:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],42:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -10875,7 +10992,9 @@ module.exports = class btcchina extends Exchange { 'countries': 'CN', 'rateLimit': 1500, 'version': 'v1', - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766368-465b3286-5ed6-11e7-9a11-0f6467e1d82b.jpg', 'api': { @@ -11196,7 +11315,7 @@ module.exports = class btcchina extends Exchange { } } -},{"./base/Exchange":8}],34:[function(require,module,exports){ +},{"./base/Exchange":8}],43:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -11213,7 +11332,9 @@ module.exports = class btcexchange extends btcturk { 'name': 'BTCExchange', 'countries': 'PH', // Philippines 'rateLimit': 1500, - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27993052-4c92911a-64aa-11e7-96d8-ec6ac3435757.jpg', 'api': 'https://www.btcexchange.ph/api', @@ -11227,7 +11348,7 @@ module.exports = class btcexchange extends btcturk { } } -},{"./btcturk.js":37}],35:[function(require,module,exports){ +},{"./btcturk.js":46}],44:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -11245,13 +11366,8 @@ module.exports = class btcmarkets extends Exchange { 'name': 'BTC Markets', 'countries': 'AU', // Australia 'rateLimit': 1000, // market data cached for 1 second (trades cached for 2 seconds) - 'hasCORS': false, - 'hasFetchOrder': true, - 'hasFetchOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchMyTrades': true, 'has': { + 'CORS': false, 'fetchOrder': true, 'fetchOrders': true, 'fetchClosedOrders': 'emulated', @@ -11616,7 +11732,7 @@ module.exports = class btcmarkets extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],36:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],45:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -11634,7 +11750,9 @@ module.exports = class btctradeua extends Exchange { 'name': 'BTC Trade UA', 'countries': 'UA', // Ukraine, 'rateLimit': 3000, - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27941483-79fc7350-62d9-11e7-9f61-ac47f28fcd96.jpg', 'api': 'https://btc-trade.com.ua/api', @@ -11961,7 +12079,7 @@ module.exports = class btctradeua extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],37:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],46:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -11979,9 +12097,11 @@ module.exports = class btcturk extends Exchange { 'name': 'BTCTurk', 'countries': 'TR', // Turkey 'rateLimit': 1000, - 'hasCORS': true, - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, + 'has': { + 'CORS': true, + 'fetchTickers': true, + 'fetchOHLCV': true, + }, 'timeframes': { '1d': '1d', }, @@ -12203,7 +12323,7 @@ module.exports = class btcturk extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],38:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],47:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -12222,7 +12342,9 @@ module.exports = class btcx extends Exchange { 'countries': [ 'IS', 'US', 'EU' ], 'rateLimit': 1500, // support in english is very poor, unable to tell rate limits 'version': 'v1', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766385-9fdcc98c-5ed6-11e7-8f14-66d5e5cd47e6.jpg', 'api': 'https://btc-x.is/api', @@ -12379,7 +12501,7 @@ module.exports = class btcx extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],39:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],48:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -12397,11 +12519,6 @@ module.exports = class bter extends Exchange { 'name': 'Bter', 'countries': [ 'VG', 'CN' ], // British Virgin Islands, China 'version': '2', - // obsolete metainfo interface - 'hasCORS': false, - 'hasFetchTickers': true, - 'hasWithdraw': true, - // new metainfo interface 'has': { 'CORS': false, 'fetchTickers': true, @@ -12684,7 +12801,7 @@ module.exports = class bter extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],40:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],49:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -12702,8 +12819,10 @@ module.exports = class bxinth extends Exchange { 'name': 'BX.in.th', 'countries': 'TH', // Thailand 'rateLimit': 1500, - 'hasCORS': false, - 'hasFetchTickers': true, + 'has': { + 'CORS': false, + 'fetchTickers': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766412-567b1eb4-5ed7-11e7-94a8-ff6a3884f6c5.jpg', 'api': 'https://bx.in.th/api', @@ -12952,7 +13071,7 @@ module.exports = class bxinth extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],41:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],50:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -12970,8 +13089,10 @@ module.exports = class ccex extends Exchange { 'name': 'C-CEX', 'countries': [ 'DE', 'EU' ], 'rateLimit': 1500, - 'hasCORS': false, - 'hasFetchTickers': true, + 'has': { + 'CORS': false, + 'fetchTickers': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766433-16881f90-5ed8-11e7-92f8-3d92cc747a6c.jpg', 'api': { @@ -13227,7 +13348,7 @@ module.exports = class ccex extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],42:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],51:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -13245,10 +13366,12 @@ module.exports = class cex extends Exchange { 'name': 'CEX.IO', 'countries': [ 'GB', 'EU', 'CY', 'RU' ], 'rateLimit': 1500, - 'hasCORS': true, - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, - 'hasFetchOpenOrders': true, + 'has': { + 'CORS': true, + 'fetchTickers': true, + 'fetchOHLCV': true, + 'fetchOpenOrders': true, + }, 'timeframes': { '1m': '1m', }, @@ -13702,7 +13825,7 @@ module.exports = class cex extends Exchange { } }; -},{"./base/Exchange":8,"./base/errors":10}],43:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],52:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -13721,8 +13844,10 @@ module.exports = class chbtc extends zb { 'countries': 'CN', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': false, - 'hasFetchOrder': true, + 'has': { + 'CORS': false, + 'fetchOrder': true + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28555659-f0040dc2-7109-11e7-9d99-688a438bf9f4.jpg', 'api': { @@ -13768,7 +13893,7 @@ module.exports = class chbtc extends zb { } -},{"./base/errors":10,"./zb.js":105}],44:[function(require,module,exports){ +},{"./base/errors":10,"./zb.js":114}],53:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -13784,7 +13909,9 @@ module.exports = class chilebit extends foxbit { 'id': 'chilebit', 'name': 'ChileBit', 'countries': 'CL', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27991414-1298f0d8-647f-11e7-9c40-d56409266336.jpg', 'api': { @@ -13798,7 +13925,7 @@ module.exports = class chilebit extends foxbit { } } -},{"./foxbit.js":57}],45:[function(require,module,exports){ +},{"./foxbit.js":66}],54:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -13816,7 +13943,9 @@ module.exports = class coincheck extends Exchange { 'name': 'coincheck', 'countries': [ 'JP', 'ID' ], 'rateLimit': 1500, - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766464-3b5c3c74-5ed9-11e7-840e-31b32968e1da.jpg', 'api': 'https://coincheck.com/api', @@ -14043,7 +14172,7 @@ module.exports = class coincheck extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],46:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],55:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -14061,13 +14190,9 @@ module.exports = class coinexchange extends Exchange { 'name': 'CoinExchange', 'countries': [ 'IN', 'JP', 'KR', 'VN', 'US' ], 'rateLimit': 1000, - // obsolete metainfo interface - 'hasPrivateAPI': false, - 'hasFetchTrades': false, - 'hasFetchCurrencies': true, - 'hasFetchTickers': true, // new metainfo interface 'has': { + 'privateAPI': false, 'fetchTrades': false, 'fetchCurrencies': true, 'fetchTickers': true, @@ -14253,7 +14378,7 @@ module.exports = class coinexchange extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],47:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],56:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -14271,7 +14396,9 @@ module.exports = class coinfloor extends Exchange { 'name': 'coinfloor', 'rateLimit': 1000, 'countries': 'UK', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28246081-623fc164-6a1c-11e7-913f-bac0d5576c90.jpg', 'api': 'https://webapi.coinfloor.co.uk:8090/bist', @@ -14445,7 +14572,7 @@ module.exports = class coinfloor extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],48:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],57:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -14463,8 +14590,10 @@ module.exports = class coingi extends Exchange { 'name': 'Coingi', 'rateLimit': 1000, 'countries': [ 'PA', 'BG', 'CN', 'US' ], // Panama, Bulgaria, China, US - 'hasFetchTickers': true, - 'hasCORS': false, + 'has': { + 'CORS': false, + 'fetchTickers': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28619707-5c9232a8-7212-11e7-86d6-98fe5d15cc6e.jpg', 'api': { @@ -14758,7 +14887,7 @@ module.exports = class coingi extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],49:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],58:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -14777,16 +14906,16 @@ module.exports = class coinmarketcap extends Exchange { 'rateLimit': 10000, 'version': 'v1', 'countries': 'US', - 'hasCORS': true, - 'hasPrivateAPI': false, - 'hasCreateOrder': false, - 'hasCancelOrder': false, - 'hasFetchBalance': false, - 'hasFetchOrderBook': false, - 'hasFetchTrades': false, - 'hasFetchTickers': true, - 'hasFetchCurrencies': true, 'has': { + 'CORS': true, + 'privateAPI': false, + 'createOrder': false, + 'cancelOrder': false, + 'fetchBalance': false, + 'fetchOrderBook': false, + 'fetchTrades': false, + 'fetchTickers': true, + 'fetchCurrencies': true, 'fetchCurrencies': true, }, 'urls': { @@ -15039,7 +15168,7 @@ module.exports = class coinmarketcap extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],50:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],59:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -15057,7 +15186,9 @@ module.exports = class coinmate extends Exchange { 'name': 'CoinMate', 'countries': [ 'GB', 'CZ', 'EU' ], // UK, Czech Republic 'rateLimit': 1000, - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27811229-c1efb510-606c-11e7-9a36-84ba2ce412d8.jpg', 'api': 'https://coinmate.io/api', @@ -15253,7 +15384,7 @@ module.exports = class coinmate extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],51:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],60:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -15272,7 +15403,9 @@ module.exports = class coinsecure extends Exchange { 'countries': 'IN', // India 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766472-9cbd200a-5ed9-11e7-9551-2267ad7bac08.jpg', 'api': 'https://api.coinsecure.in', @@ -15581,7 +15714,7 @@ module.exports = class coinsecure extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],52:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],61:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -15599,7 +15732,9 @@ module.exports = class coinspot extends Exchange { 'name': 'CoinSpot', 'countries': 'AU', // Australia 'rateLimit': 1000, - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28208429-3cacdf9a-6896-11e7-854e-4c79a772a30f.jpg', 'api': { @@ -15743,7 +15878,7 @@ module.exports = class coinspot extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],53:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],62:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -15761,19 +15896,8 @@ module.exports = class cryptopia extends Exchange { 'name': 'Cryptopia', 'rateLimit': 1500, 'countries': 'NZ', // New Zealand - 'hasCORS': false, - // obsolete metainfo interface - 'hasFetchTickers': true, - 'hasFetchOrder': true, - 'hasFetchOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchMyTrades': true, - 'hasFetchCurrencies': true, - 'hasDeposit': true, - 'hasWithdraw': true, - // new metainfo interface 'has': { + 'CORS': false, 'fetchTickers': true, 'fetchOrder': 'emulated', 'fetchOrders': 'emulated', @@ -16378,7 +16502,7 @@ module.exports = class cryptopia extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],54:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],63:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -16395,13 +16519,15 @@ module.exports = class dsx extends liqui { 'name': 'DSX', 'countries': 'UK', 'rateLimit': 1500, - 'hasCORS': false, - 'hasFetchOrder': true, - 'hasFetchOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchTickers': true, - 'hasFetchMyTrades': true, + 'has': { + 'CORS': false, + 'fetchOrder': true, + 'fetchOrders': true, + 'fetchOpenOrders': true, + 'fetchClosedOrders': true, + 'fetchTickers': true, + 'fetchMyTrades': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27990275-1413158a-645a-11e7-931c-94717f7510e3.jpg', 'api': { @@ -16527,7 +16653,7 @@ module.exports = class dsx extends liqui { } } -},{"./liqui.js":77}],55:[function(require,module,exports){ +},{"./liqui.js":86}],64:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -16546,9 +16672,11 @@ module.exports = class exmo extends Exchange { 'countries': [ 'ES', 'RU' ], // Spain, Russia 'rateLimit': 1000, // once every 350 ms ≈ 180 requests per minute ≈ 3 requests per second 'version': 'v1', - 'hasCORS': false, - 'hasFetchTickers': true, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'fetchTickers': true, + 'withdraw': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766491-1b0ea956-5eda-11e7-9225-40d67b481b8d.jpg', 'api': 'https://api.exmo.com', @@ -16829,7 +16957,7 @@ module.exports = class exmo extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],56:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],65:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -16848,7 +16976,9 @@ module.exports = class flowbtc extends Exchange { 'countries': 'BR', // Brazil 'version': 'v1', 'rateLimit': 1000, - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28162465-cd815d4c-67cf-11e7-8e57-438bea0523a2.jpg', 'api': 'https://api.flowbtc.com:8400/ajax', @@ -17056,7 +17186,7 @@ module.exports = class flowbtc extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],57:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],66:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -17073,7 +17203,9 @@ module.exports = class foxbit extends Exchange { 'id': 'foxbit', 'name': 'FoxBit', 'countries': 'BR', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'rateLimit': 1000, 'version': 'v1', 'urls': { @@ -17250,7 +17382,7 @@ module.exports = class foxbit extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],58:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],67:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -17267,7 +17399,9 @@ module.exports = class fybse extends Exchange { 'id': 'fybse', 'name': 'FYB-SE', 'countries': 'SE', // Sweden - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'rateLimit': 1500, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766512-31019772-5edb-11e7-8241-2e675e6797f1.jpg', @@ -17425,7 +17559,7 @@ module.exports = class fybse extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],59:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],68:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -17441,7 +17575,9 @@ module.exports = class fybsg extends fybse { 'id': 'fybsg', 'name': 'FYB-SG', 'countries': 'SG', // Singapore - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766513-3364d56a-5edb-11e7-9e6b-d5898bb89c81.jpg', 'api': 'https://www.fybsg.com/api/SGD', @@ -17455,7 +17591,7 @@ module.exports = class fybsg extends fybse { } } -},{"./fybse.js":58}],60:[function(require,module,exports){ +},{"./fybse.js":67}],69:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -17474,9 +17610,11 @@ module.exports = class gatecoin extends Exchange { 'rateLimit': 2000, 'countries': 'HK', // Hong Kong 'comment': 'a regulated/licensed exchange', - 'hasCORS': false, - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, + 'has': { + 'CORS': false, + 'fetchTickers': true, + 'fetchOHLCV': true, + }, 'timeframes': { '1m': '1m', '15m': '15m', @@ -17872,7 +18010,7 @@ module.exports = class gatecoin extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],61:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],70:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -17919,7 +18057,7 @@ module.exports = class gateio extends bter { } } -},{"./bter.js":39}],62:[function(require,module,exports){ +},{"./bter.js":48}],71:[function(require,module,exports){ 'use strict'; // ---------------------------------------------------------------------------- @@ -17938,16 +18076,6 @@ module.exports = class gdax extends Exchange { 'countries': 'US', 'rateLimit': 1000, 'userAgent': this.userAgents['chrome'], - // obsolete metainfo interface - 'hasCORS': true, - 'hasFetchOHLCV': true, - 'hasDeposit': true, - 'hasWithdraw': true, - 'hasFetchOrder': true, - 'hasFetchOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - // new metainfo interface 'has': { 'CORS': true, 'fetchOHLCV': true, @@ -18480,7 +18608,7 @@ module.exports = class gdax extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],63:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],72:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -18499,10 +18627,6 @@ module.exports = class gemini extends Exchange { 'countries': 'US', 'rateLimit': 1500, // 200 for private API 'version': 'v1', - // obsolete metainfo interface - 'hasCORS': false, - 'hasWithdraw': true, - // new metainfo interface 'has': { 'CORS': false, 'withdraw': true, @@ -18730,7 +18854,7 @@ module.exports = class gemini extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],64:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],73:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -18768,7 +18892,7 @@ module.exports = class getbtc extends _1btcxe { } } -},{"./_1btcxe.js":4}],65:[function(require,module,exports){ +},{"./_1btcxe.js":4}],74:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -18787,12 +18911,13 @@ module.exports = class hitbtc extends Exchange { 'countries': 'UK', 'rateLimit': 1500, 'version': '1', - 'hasCORS': false, - 'hasFetchTickers': true, - 'hasFetchOrder': true, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'fetchOrder': true, + 'fetchOpenOrders': true, + 'fetchClosedOrders': true, + 'withdraw': true + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766555-8eaec20e-5edc-11e7-9c5b-6dc69fc42f5e.jpg', 'api': 'http://api.hitbtc.com', @@ -19642,7 +19767,7 @@ module.exports = class hitbtc extends Exchange { } }; -},{"./base/Exchange":8,"./base/errors":10}],66:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],75:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -19661,19 +19786,8 @@ module.exports = class hitbtc2 extends hitbtc { 'countries': 'UK', 'rateLimit': 1500, 'version': '2', - 'hasCORS': true, - // older metainfo interface - 'hasFetchOHLCV': true, - 'hasFetchTickers': true, - 'hasFetchOrder': true, - 'hasFetchOrders': false, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchMyTrades': true, - 'hasWithdraw': true, - 'hasFetchCurrencies': true, - // new metainfo interface 'has': { + 'CORS': true, 'fetchCurrencies': true, 'fetchOHLCV': true, 'fetchTickers': true, @@ -20724,7 +20838,7 @@ module.exports = class hitbtc2 extends hitbtc { } } -},{"./base/errors":10,"./hitbtc":65}],67:[function(require,module,exports){ +},{"./base/errors":10,"./hitbtc":74}],76:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -20743,8 +20857,10 @@ module.exports = class huobi extends Exchange { 'countries': 'CN', 'rateLimit': 2000, 'version': 'v3', - 'hasCORS': false, - 'hasFetchOHLCV': true, + 'has': { + 'CORS': false, + 'fetchOHLCV': true, + }, 'timeframes': { '1m': '001', '5m': '005', @@ -20981,7 +21097,7 @@ module.exports = class huobi extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],68:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],77:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -20997,7 +21113,9 @@ module.exports = class huobicny extends huobipro { 'id': 'huobicny', 'name': 'Huobi CNY', 'hostname': 'be.huobi.com', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766569-15aa7b9a-5edd-11e7-9e7f-44791f4ee49c.jpg', 'api': 'https://be.huobi.com', @@ -21008,7 +21126,7 @@ module.exports = class huobicny extends huobipro { } } -},{"./huobipro.js":69}],69:[function(require,module,exports){ +},{"./huobipro.js":78}],78:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -21031,13 +21149,8 @@ module.exports = class huobipro extends Exchange { 'accounts': undefined, 'accountsById': undefined, 'hostname': 'api.huobi.pro', - 'hasCORS': false, - // obsolete metainfo structure - 'hasFetchOHLCV': true, - 'hasFetchOrders': true, - 'hasFetchOpenOrders': true, - // new metainfo structure 'has': { + 'CORS': false, 'fetchOHCLV': true, 'fetchOrders': true, 'fetchOpenOrders': true, @@ -21494,7 +21607,7 @@ module.exports = class huobipro extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],70:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],79:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -21511,7 +21624,9 @@ module.exports = class independentreserve extends Exchange { 'name': 'Independent Reserve', 'countries': [ 'AU', 'NZ' ], // Australia, New Zealand 'rateLimit': 1000, - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30521662-cf3f477c-9bcb-11e7-89bc-d1ac85012eda.jpg', 'api': { @@ -21751,7 +21866,7 @@ module.exports = class independentreserve extends Exchange { } } -},{"./base/Exchange":8}],71:[function(require,module,exports){ +},{"./base/Exchange":8}],80:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -21770,7 +21885,9 @@ module.exports = class itbit extends Exchange { 'countries': 'US', 'rateLimit': 2000, 'version': 'v1', - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27822159-66153620-60ad-11e7-89e7-005f6d7f3de0.jpg', 'api': 'https://api.itbit.com', @@ -21986,7 +22103,7 @@ module.exports = class itbit extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],72:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],81:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -22004,8 +22121,10 @@ module.exports = class jubi extends btcbox { 'countries': 'CN', 'rateLimit': 1500, 'version': 'v1', - 'hasCORS': false, - 'hasFetchTickers': true, + 'has': { + 'CORS': false, + 'fetchTickers': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766581-9d397d9a-5edd-11e7-8fb9-5d8236c0e692.jpg', 'api': 'https://www.jubi.com/api', @@ -22038,7 +22157,7 @@ module.exports = class jubi extends btcbox { } } -},{"./btcbox.js":32}],73:[function(require,module,exports){ +},{"./btcbox.js":41}],82:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -22057,18 +22176,8 @@ module.exports = class kraken extends Exchange { 'countries': 'US', 'version': '0', 'rateLimit': 3000, - 'hasCORS': false, - // obsolete metainfo interface - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, - 'hasFetchOrder': true, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchMyTrades': true, - 'hasWithdraw': true, - 'hasFetchCurrencies': true, - // new metainfo interface 'has': { + 'CORS': false, 'fetchCurrencies': true, 'fetchTickers': true, 'fetchOHLCV': true, @@ -22846,7 +22955,7 @@ module.exports = class kraken extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],74:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],83:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -22865,20 +22974,9 @@ module.exports = class kucoin extends Exchange { 'countries': 'HK', // Hong Kong 'version': 'v1', 'rateLimit': 2000, - 'hasCORS': false, 'userAgent': this.userAgents['chrome'], - // obsolete metainfo interface - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, - 'hasFetchOrder': false, - 'hasFetchOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchMyTrades': false, - 'hasFetchCurrencies': true, - 'hasWithdraw': true, - // new metainfo interface 'has': { + 'CORS': false, 'fetchTickers': true, 'fetchOHLCV': true, // see the method implementation below 'fetchOrder': false, @@ -23477,7 +23575,7 @@ module.exports = class kucoin extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],75:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],84:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -23496,9 +23594,11 @@ module.exports = class kuna extends acx { 'countries': 'UA', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': false, - 'hasFetchTickers': false, - 'hasFetchOHLCV': false, + 'has': { + 'CORS': false, + 'fetchTickers': false, + 'fetchOHLCV': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/31697638-912824fa-b3c1-11e7-8c36-cf9606eb94ac.jpg', 'api': 'https://kuna.io', @@ -23672,7 +23772,7 @@ module.exports = class kuna extends acx { } }; -},{"./acx.js":5,"./base/errors":10}],76:[function(require,module,exports){ +},{"./acx.js":5,"./base/errors":10}],85:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -23690,7 +23790,9 @@ module.exports = class lakebtc extends Exchange { 'name': 'LakeBTC', 'countries': 'US', 'version': 'api_v2', - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28074120-72b7c38a-6660-11e7-92d9-d9027502281d.jpg', 'api': 'https://api.lakebtc.com', @@ -23905,7 +24007,7 @@ module.exports = class lakebtc extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],77:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],86:[function(require,module,exports){ 'use strict'; const Exchange = require ('./base/Exchange'); @@ -23919,18 +24021,9 @@ module.exports = class liqui extends Exchange { 'countries': 'UA', 'rateLimit': 3000, 'version': '3', - 'hasCORS': false, 'userAgent': this.userAgents['chrome'], - // obsolete metainfo interface - 'hasFetchOrder': true, - 'hasFetchOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchTickers': true, - 'hasFetchMyTrades': true, - 'hasWithdraw': true, - // new metainfo interface 'has': { + 'CORS': false, 'fetchOrder': true, 'fetchOrders': 'emulated', 'fetchOpenOrders': true, @@ -24594,7 +24687,7 @@ module.exports = class liqui extends Exchange { } }; -},{"./base/Exchange":8,"./base/errors":10}],78:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],87:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -24612,12 +24705,8 @@ module.exports = class livecoin extends Exchange { 'name': 'LiveCoin', 'countries': [ 'US', 'UK', 'RU' ], 'rateLimit': 1000, - 'hasCORS': false, - // obsolete metainfo interface - 'hasFetchTickers': true, - 'hasFetchCurrencies': true, - // new metainfo interface 'has': { + 'CORS': false, 'fetchTickers': true, 'fetchCurrencies': true, }, @@ -25162,7 +25251,7 @@ module.exports = class livecoin extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],79:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],88:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -25181,10 +25270,8 @@ module.exports = class luno extends Exchange { 'countries': [ 'GB', 'SG', 'ZA' ], 'rateLimit': 10000, 'version': '1', - 'hasCORS': false, - 'hasFetchTickers': true, - 'hasFetchOrder': true, 'has': { + 'CORS': false, 'fetchTickers': true, 'fetchOrder': true, }, @@ -25472,7 +25559,7 @@ module.exports = class luno extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],80:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],89:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -25490,12 +25577,8 @@ module.exports = class lykke extends Exchange { 'countries': 'CH', 'version': 'v1', 'rateLimit': 200, - 'hasCORS': false, - // obsolete metainfo interface - 'hasFetchTrades': false, - 'hasFetchOHLCV': false, - // new metainfo interface 'has': { + 'CORS': false, 'fetchOHLCV': false, 'fetchTrades': false, }, @@ -25845,7 +25928,7 @@ module.exports = class lykke extends Exchange { } } -},{"./base/Exchange":8}],81:[function(require,module,exports){ +},{"./base/Exchange":8}],90:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -25864,8 +25947,10 @@ module.exports = class mercado extends Exchange { 'countries': 'BR', // Brazil 'rateLimit': 1000, 'version': 'v3', - 'hasCORS': true, - 'hasWithdraw': true, + 'has': { + 'CORS': true, + 'withdraw': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27837060-e7c58714-60ea-11e7-9192-f05e86adb83f.jpg', 'api': { @@ -26140,7 +26225,7 @@ module.exports = class mercado extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],82:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],91:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -26159,7 +26244,9 @@ module.exports = class mixcoins extends Exchange { 'countries': [ 'GB', 'HK' ], 'rateLimit': 1500, 'version': 'v1', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30237212-ed29303c-9535-11e7-8af8-fcd381cfa20c.jpg', 'api': 'https://mixcoins.com/api', @@ -26325,7 +26412,7 @@ module.exports = class mixcoins extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],83:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],92:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -26344,7 +26431,9 @@ module.exports = class nova extends Exchange { 'countries': 'TZ', // Tanzania 'rateLimit': 2000, 'version': 'v2', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30518571-78ca0bca-9b8a-11e7-8840-64b83a4a94b2.jpg', 'api': 'https://novaexchange.com/remote', @@ -26553,7 +26642,7 @@ module.exports = class nova extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],84:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],93:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -26569,7 +26658,9 @@ module.exports = class okcoincny extends okcoinusd { 'id': 'okcoincny', 'name': 'OKCoin CNY', 'countries': 'CN', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766792-8be9157a-5ee5-11e7-926c-6d69b8d3378d.jpg', 'api': { @@ -26591,7 +26682,7 @@ module.exports = class okcoincny extends okcoinusd { } } -},{"./okcoinusd.js":85}],85:[function(require,module,exports){ +},{"./okcoinusd.js":94}],94:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -26608,27 +26699,19 @@ module.exports = class okcoinusd extends Exchange { 'id': 'okcoinusd', 'name': 'OKCoin USD', 'countries': [ 'CN', 'US' ], - 'hasCORS': false, 'version': 'v1', 'rateLimit': 1000, // up to 3000 requests per 5 minutes ≈ 600 requests per minute ≈ 10 requests per second ≈ 100 ms - // obsolete metainfo interface - 'hasFetchOHLCV': true, - 'hasFetchOrder': true, - 'hasFetchOrders': false, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasWithdraw': true, - // new metainfo interface 'has': { + 'CORS': false, 'fetchOHLCV': true, 'fetchOrder': true, 'fetchOrders': false, 'fetchOpenOrders': true, 'fetchClosedOrders': true, 'withdraw': true, + 'futureMarkets': false, }, 'extension': '.do', // appended to endpoint URL - 'hasFutureMarkets': false, 'timeframes': { '1m': '1min', '3m': '3min', @@ -27248,7 +27331,7 @@ module.exports = class okcoinusd extends Exchange { } }; -},{"./base/Exchange":8,"./base/errors":10}],86:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],95:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -27264,8 +27347,10 @@ module.exports = class okex extends okcoinusd { 'id': 'okex', 'name': 'OKEX', 'countries': [ 'CN', 'US' ], - 'hasCORS': false, - 'hasFutureMarkets': true, + 'has': { + 'CORS': false, + 'futureMarkets': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/32552768-0d6dd3c6-c4a6-11e7-90f8-c043b64756a7.jpg', 'api': { @@ -27281,7 +27366,7 @@ module.exports = class okex extends okcoinusd { } } -},{"./okcoinusd.js":85}],87:[function(require,module,exports){ +},{"./okcoinusd.js":94}],96:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -27300,7 +27385,9 @@ module.exports = class paymium extends Exchange { 'countries': [ 'FR', 'EU' ], 'rateLimit': 2000, 'version': 'v1', - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27790564-a945a9d4-5ff9-11e7-9d2d-b635763f2f24.jpg', 'api': 'https://paymium.com/api', @@ -27491,7 +27578,7 @@ module.exports = class paymium extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],88:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],97:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -27509,19 +27596,8 @@ module.exports = class poloniex extends Exchange { 'name': 'Poloniex', 'countries': 'US', 'rateLimit': 1000, // up to 6 calls per second - 'hasCORS': true, - // obsolete metainfo interface - 'hasFetchMyTrades': true, - 'hasFetchOrder': true, - 'hasFetchOrders': true, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasFetchTickers': true, - 'hasFetchCurrencies': true, - 'hasWithdraw': true, - 'hasFetchOHLCV': true, - // new metainfo interface 'has': { + 'CORS': true, 'fetchOHLCV': true, 'fetchMyTrades': true, 'fetchOrder': 'emulated', @@ -28306,7 +28382,7 @@ module.exports = class poloniex extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],89:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],98:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -28325,9 +28401,9 @@ module.exports = class qryptos extends Exchange { 'countries': [ 'CN', 'TW' ], 'version': '2', 'rateLimit': 1000, - 'hasFetchTickers': true, - 'hasCORS': false, 'has': { + 'CORS': false, + 'fetchTickers': true, 'fetchOrder': true, 'fetchOrders': true, 'fetchOpenOrders': true, @@ -28710,7 +28786,7 @@ module.exports = class qryptos extends Exchange { } }; -},{"./base/Exchange":8,"./base/errors":10}],90:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],99:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -28729,11 +28805,8 @@ module.exports = class quadrigacx extends Exchange { 'countries': 'CA', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': true, - // obsolete metainfo interface - 'hasWithdraw': true, - // new metainfo interface 'has': { + 'CORS': true, 'withdraw': true, }, 'urls': { @@ -28956,7 +29029,7 @@ module.exports = class quadrigacx extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],91:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],100:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -28974,8 +29047,10 @@ module.exports = class quoinex extends qryptos { 'countries': [ 'JP', 'SG', 'VN' ], 'version': '2', 'rateLimit': 1000, - 'hasFetchTickers': true, - 'hasCORS': false, + 'has': { + 'CORS': false, + 'fetchTickers': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/35047114-0e24ad4a-fbaa-11e7-96a9-69c1a756083b.jpg', 'api': 'https://api.quoine.com', @@ -28990,7 +29065,7 @@ module.exports = class quoinex extends qryptos { } }; -},{"./qryptos.js":89}],92:[function(require,module,exports){ +},{"./qryptos.js":98}],101:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -29008,9 +29083,11 @@ module.exports = class southxchange extends Exchange { 'name': 'SouthXchange', 'countries': 'AR', // Argentina 'rateLimit': 1000, - 'hasFetchTickers': true, - 'hasCORS': false, - 'hasWithdraw': true, + 'has': { + 'CORS': true, + 'fetchTickers': true, + 'withdraw': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27838912-4f94ec8a-60f6-11e7-9e5d-bbf9bd50a559.jpg', 'api': 'https://www.southxchange.com/api', @@ -29244,7 +29321,7 @@ module.exports = class southxchange extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],93:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],102:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -29260,7 +29337,9 @@ module.exports = class surbitcoin extends foxbit { 'id': 'surbitcoin', 'name': 'SurBitcoin', 'countries': 'VE', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27991511-f0a50194-6481-11e7-99b5-8f02932424cc.jpg', 'api': { @@ -29274,7 +29353,7 @@ module.exports = class surbitcoin extends foxbit { } } -},{"./foxbit.js":57}],94:[function(require,module,exports){ +},{"./foxbit.js":66}],103:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -29293,8 +29372,10 @@ module.exports = class therock extends Exchange { 'countries': 'MT', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': false, - 'hasFetchTickers': true, + 'has': { + 'CORS': false, + 'fetchTickers': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766869-75057fa2-5ee9-11e7-9a6f-13e641fa4707.jpg', 'api': 'https://api.therocktrading.com', @@ -29553,7 +29634,7 @@ module.exports = class therock extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],95:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],104:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -29571,8 +29652,10 @@ module.exports = class tidex extends liqui { 'countries': 'UK', 'rateLimit': 2000, 'version': '3', - // 'hasCORS': false, - // 'hasFetchTickers': true, + 'has': { + // 'CORS': false, + // 'fetchTickers': true + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30781780-03149dc4-a12e-11e7-82bb-313b269d24d4.jpg', 'api': { @@ -29702,7 +29785,7 @@ module.exports = class tidex extends liqui { } } -},{"./liqui.js":77}],96:[function(require,module,exports){ +},{"./liqui.js":86}],105:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -29718,7 +29801,9 @@ module.exports = class urdubit extends foxbit { 'id': 'urdubit', 'name': 'UrduBit', 'countries': 'PK', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27991453-156bf3ae-6480-11e7-82eb-7295fe1b5bb4.jpg', 'api': { @@ -29732,7 +29817,7 @@ module.exports = class urdubit extends foxbit { } } -},{"./foxbit.js":57}],97:[function(require,module,exports){ +},{"./foxbit.js":66}],106:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -29750,7 +29835,9 @@ module.exports = class vaultoro extends Exchange { 'countries': 'CH', 'rateLimit': 1000, 'version': '1', - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766880-f205e870-5ee9-11e7-8fe2-0d5b15880752.jpg', 'api': 'https://api.vaultoro.com', @@ -29944,7 +30031,7 @@ module.exports = class vaultoro extends Exchange { } } -},{"./base/Exchange":8}],98:[function(require,module,exports){ +},{"./base/Exchange":8}],107:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -29960,7 +30047,9 @@ module.exports = class vbtc extends foxbit { 'id': 'vbtc', 'name': 'VBTC', 'countries': 'VN', - 'hasCORS': false, + 'has': { + 'CORS': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27991481-1f53d1d8-6481-11e7-884e-21d17e7939db.jpg', 'api': { @@ -29974,7 +30063,7 @@ module.exports = class vbtc extends foxbit { } } -},{"./foxbit.js":57}],99:[function(require,module,exports){ +},{"./foxbit.js":66}],108:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -29992,7 +30081,9 @@ module.exports = class virwox extends Exchange { 'name': 'VirWoX', 'countries': [ 'AT', 'EU' ], 'rateLimit': 1000, - 'hasCORS': true, + 'has': { + 'CORS': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766894-6da9d360-5eea-11e7-90aa-41f2711b7405.jpg', 'api': { @@ -30259,7 +30350,7 @@ module.exports = class virwox extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],100:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],109:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -30277,8 +30368,10 @@ module.exports = class wex extends liqui { 'name': 'WEX', 'countries': 'NZ', // New Zealand 'version': '3', - 'hasFetchTickers': true, - 'hasCORS': false, + 'has': { + 'CORS': false, + 'fetchTickers': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30652751-d74ec8f8-9e31-11e7-98c5-71469fcef03e.jpg', 'api': { @@ -30402,7 +30495,7 @@ module.exports = class wex extends liqui { } } -},{"./base/errors":10,"./liqui.js":77}],101:[function(require,module,exports){ +},{"./base/errors":10,"./liqui.js":86}],110:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -30421,10 +30514,12 @@ module.exports = class xbtce extends Exchange { 'countries': 'RU', 'rateLimit': 2000, // responses are cached every 2 seconds 'version': 'v1', - 'hasPublicAPI': false, - 'hasCORS': false, - 'hasFetchTickers': true, - 'hasFetchOHLCV': false, + 'has': { + 'publicAPI': false, + 'CORS': false, + 'fetchTickers': true, + 'fetchOHLCV': false, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28059414-e235970c-662c-11e7-8c3a-08e31f78684b.jpg', 'api': 'https://cryptottlivewebapi.xbtce.net:8443/api', @@ -30745,7 +30840,7 @@ module.exports = class xbtce extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],102:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],111:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -30764,8 +30859,10 @@ module.exports = class yobit extends liqui { 'countries': 'RU', 'rateLimit': 3000, // responses are cached every 2 seconds 'version': '3', - 'hasCORS': false, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'withdraw': true + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766910-cdcbfdae-5eea-11e7-9859-03fea873272d.jpg', 'api': { @@ -30948,7 +31045,7 @@ module.exports = class yobit extends liqui { } -},{"./base/errors":10,"./liqui.js":77}],103:[function(require,module,exports){ +},{"./base/errors":10,"./liqui.js":86}],112:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -30966,9 +31063,11 @@ module.exports = class yunbi extends acx { 'countries': 'CN', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': false, - 'hasFetchTickers': true, - 'hasFetchOHLCV': true, + 'has': { + 'CORS': false, + 'fetchTickers': true, + 'fetchOHLCV': true, + }, 'timeframes': { '1m': '1', '5m': '5', @@ -31030,7 +31129,7 @@ module.exports = class yunbi extends acx { } } -},{"./acx.js":5}],104:[function(require,module,exports){ +},{"./acx.js":5}],113:[function(require,module,exports){ 'use strict'; // --------------------------------------------------------------------------- @@ -31049,10 +31148,12 @@ module.exports = class zaif extends Exchange { 'countries': 'JP', 'rateLimit': 2000, 'version': '1', - 'hasCORS': false, - 'hasFetchOpenOrders': true, - 'hasFetchClosedOrders': true, - 'hasWithdraw': true, + 'has': { + 'CORS': false, + 'fetchOpenOrders': true, + 'fetchClosedOrders': true, + 'withdraw': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766927-39ca2ada-5eeb-11e7-972f-1b4199518ca6.jpg', 'api': 'https://api.zaif.jp', @@ -31390,7 +31491,7 @@ module.exports = class zaif extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],105:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],114:[function(require,module,exports){ "use strict"; // --------------------------------------------------------------------------- @@ -31409,8 +31510,10 @@ module.exports = class zb extends Exchange { 'countries': 'CN', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': false, - 'hasFetchOrder': true, + 'has': { + 'CORS': false, + 'fetchOrder': true, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/32859187-cd5214f0-ca5e-11e7-967d-96568e2e2bd1.jpg', 'api': { @@ -31728,7 +31831,7 @@ module.exports = class zb extends Exchange { } } -},{"./base/Exchange":8,"./base/errors":10}],106:[function(require,module,exports){ +},{"./base/Exchange":8,"./base/errors":10}],115:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -31961,7 +32064,7 @@ module.exports = class zb extends Exchange { return CryptoJS.AES; })); -},{"./cipher-core":107,"./core":108,"./enc-base64":109,"./evpkdf":111,"./md5":116}],107:[function(require,module,exports){ +},{"./cipher-core":116,"./core":117,"./enc-base64":118,"./evpkdf":120,"./md5":125}],116:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -32842,7 +32945,7 @@ module.exports = class zb extends Exchange { })); -},{"./core":108,"./evpkdf":111}],108:[function(require,module,exports){ +},{"./core":117,"./evpkdf":120}],117:[function(require,module,exports){ ;(function (root, factory) { if (typeof exports === "object") { // CommonJS @@ -33603,7 +33706,7 @@ module.exports = class zb extends Exchange { return CryptoJS; })); -},{}],109:[function(require,module,exports){ +},{}],118:[function(require,module,exports){ ;(function (root, factory) { if (typeof exports === "object") { // CommonJS @@ -33739,7 +33842,7 @@ module.exports = class zb extends Exchange { return CryptoJS.enc.Base64; })); -},{"./core":108}],110:[function(require,module,exports){ +},{"./core":117}],119:[function(require,module,exports){ ;(function (root, factory) { if (typeof exports === "object") { // CommonJS @@ -33889,7 +33992,7 @@ module.exports = class zb extends Exchange { return CryptoJS.enc.Utf16; })); -},{"./core":108}],111:[function(require,module,exports){ +},{"./core":117}],120:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -34022,7 +34125,7 @@ module.exports = class zb extends Exchange { return CryptoJS.EvpKDF; })); -},{"./core":108,"./hmac":113,"./sha1":132}],112:[function(require,module,exports){ +},{"./core":117,"./hmac":122,"./sha1":141}],121:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -34089,7 +34192,7 @@ module.exports = class zb extends Exchange { return CryptoJS.format.Hex; })); -},{"./cipher-core":107,"./core":108}],113:[function(require,module,exports){ +},{"./cipher-core":116,"./core":117}],122:[function(require,module,exports){ ;(function (root, factory) { if (typeof exports === "object") { // CommonJS @@ -34233,7 +34336,7 @@ module.exports = class zb extends Exchange { })); -},{"./core":108}],114:[function(require,module,exports){ +},{"./core":117}],123:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -34252,7 +34355,7 @@ module.exports = class zb extends Exchange { return CryptoJS; })); -},{"./aes":106,"./cipher-core":107,"./core":108,"./enc-base64":109,"./enc-utf16":110,"./evpkdf":111,"./format-hex":112,"./hmac":113,"./lib-typedarrays":115,"./md5":116,"./mode-cfb":117,"./mode-ctr":119,"./mode-ctr-gladman":118,"./mode-ecb":120,"./mode-ofb":121,"./pad-ansix923":122,"./pad-iso10126":123,"./pad-iso97971":124,"./pad-nopadding":125,"./pad-zeropadding":126,"./pbkdf2":127,"./rabbit":129,"./rabbit-legacy":128,"./rc4":130,"./ripemd160":131,"./sha1":132,"./sha224":133,"./sha256":134,"./sha3":135,"./sha384":136,"./sha512":137,"./tripledes":138,"./x64-core":139}],115:[function(require,module,exports){ +},{"./aes":115,"./cipher-core":116,"./core":117,"./enc-base64":118,"./enc-utf16":119,"./evpkdf":120,"./format-hex":121,"./hmac":122,"./lib-typedarrays":124,"./md5":125,"./mode-cfb":126,"./mode-ctr":128,"./mode-ctr-gladman":127,"./mode-ecb":129,"./mode-ofb":130,"./pad-ansix923":131,"./pad-iso10126":132,"./pad-iso97971":133,"./pad-nopadding":134,"./pad-zeropadding":135,"./pbkdf2":136,"./rabbit":138,"./rabbit-legacy":137,"./rc4":139,"./ripemd160":140,"./sha1":141,"./sha224":142,"./sha256":143,"./sha3":144,"./sha384":145,"./sha512":146,"./tripledes":147,"./x64-core":148}],124:[function(require,module,exports){ ;(function (root, factory) { if (typeof exports === "object") { // CommonJS @@ -34329,7 +34432,7 @@ module.exports = class zb extends Exchange { return CryptoJS.lib.WordArray; })); -},{"./core":108}],116:[function(require,module,exports){ +},{"./core":117}],125:[function(require,module,exports){ ;(function (root, factory) { if (typeof exports === "object") { // CommonJS @@ -34598,7 +34701,7 @@ module.exports = class zb extends Exchange { return CryptoJS.MD5; })); -},{"./core":108}],117:[function(require,module,exports){ +},{"./core":117}],126:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -34677,7 +34780,7 @@ module.exports = class zb extends Exchange { return CryptoJS.mode.CFB; })); -},{"./cipher-core":107,"./core":108}],118:[function(require,module,exports){ +},{"./cipher-core":116,"./core":117}],127:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -34794,7 +34897,7 @@ module.exports = class zb extends Exchange { return CryptoJS.mode.CTRGladman; })); -},{"./cipher-core":107,"./core":108}],119:[function(require,module,exports){ +},{"./cipher-core":116,"./core":117}],128:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -34853,7 +34956,7 @@ module.exports = class zb extends Exchange { return CryptoJS.mode.CTR; })); -},{"./cipher-core":107,"./core":108}],120:[function(require,module,exports){ +},{"./cipher-core":116,"./core":117}],129:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -34894,7 +34997,7 @@ module.exports = class zb extends Exchange { return CryptoJS.mode.ECB; })); -},{"./cipher-core":107,"./core":108}],121:[function(require,module,exports){ +},{"./cipher-core":116,"./core":117}],130:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -34949,7 +35052,7 @@ module.exports = class zb extends Exchange { return CryptoJS.mode.OFB; })); -},{"./cipher-core":107,"./core":108}],122:[function(require,module,exports){ +},{"./cipher-core":116,"./core":117}],131:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -34999,7 +35102,7 @@ module.exports = class zb extends Exchange { return CryptoJS.pad.Ansix923; })); -},{"./cipher-core":107,"./core":108}],123:[function(require,module,exports){ +},{"./cipher-core":116,"./core":117}],132:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -35044,7 +35147,7 @@ module.exports = class zb extends Exchange { return CryptoJS.pad.Iso10126; })); -},{"./cipher-core":107,"./core":108}],124:[function(require,module,exports){ +},{"./cipher-core":116,"./core":117}],133:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -35085,7 +35188,7 @@ module.exports = class zb extends Exchange { return CryptoJS.pad.Iso97971; })); -},{"./cipher-core":107,"./core":108}],125:[function(require,module,exports){ +},{"./cipher-core":116,"./core":117}],134:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -35116,7 +35219,7 @@ module.exports = class zb extends Exchange { return CryptoJS.pad.NoPadding; })); -},{"./cipher-core":107,"./core":108}],126:[function(require,module,exports){ +},{"./cipher-core":116,"./core":117}],135:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -35162,7 +35265,7 @@ module.exports = class zb extends Exchange { return CryptoJS.pad.ZeroPadding; })); -},{"./cipher-core":107,"./core":108}],127:[function(require,module,exports){ +},{"./cipher-core":116,"./core":117}],136:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -35308,7 +35411,7 @@ module.exports = class zb extends Exchange { return CryptoJS.PBKDF2; })); -},{"./core":108,"./hmac":113,"./sha1":132}],128:[function(require,module,exports){ +},{"./core":117,"./hmac":122,"./sha1":141}],137:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -35499,7 +35602,7 @@ module.exports = class zb extends Exchange { return CryptoJS.RabbitLegacy; })); -},{"./cipher-core":107,"./core":108,"./enc-base64":109,"./evpkdf":111,"./md5":116}],129:[function(require,module,exports){ +},{"./cipher-core":116,"./core":117,"./enc-base64":118,"./evpkdf":120,"./md5":125}],138:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -35692,7 +35795,7 @@ module.exports = class zb extends Exchange { return CryptoJS.Rabbit; })); -},{"./cipher-core":107,"./core":108,"./enc-base64":109,"./evpkdf":111,"./md5":116}],130:[function(require,module,exports){ +},{"./cipher-core":116,"./core":117,"./enc-base64":118,"./evpkdf":120,"./md5":125}],139:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -35832,7 +35935,7 @@ module.exports = class zb extends Exchange { return CryptoJS.RC4; })); -},{"./cipher-core":107,"./core":108,"./enc-base64":109,"./evpkdf":111,"./md5":116}],131:[function(require,module,exports){ +},{"./cipher-core":116,"./core":117,"./enc-base64":118,"./evpkdf":120,"./md5":125}],140:[function(require,module,exports){ ;(function (root, factory) { if (typeof exports === "object") { // CommonJS @@ -36100,7 +36203,7 @@ module.exports = class zb extends Exchange { return CryptoJS.RIPEMD160; })); -},{"./core":108}],132:[function(require,module,exports){ +},{"./core":117}],141:[function(require,module,exports){ ;(function (root, factory) { if (typeof exports === "object") { // CommonJS @@ -36251,7 +36354,7 @@ module.exports = class zb extends Exchange { return CryptoJS.SHA1; })); -},{"./core":108}],133:[function(require,module,exports){ +},{"./core":117}],142:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -36332,7 +36435,7 @@ module.exports = class zb extends Exchange { return CryptoJS.SHA224; })); -},{"./core":108,"./sha256":134}],134:[function(require,module,exports){ +},{"./core":117,"./sha256":143}],143:[function(require,module,exports){ ;(function (root, factory) { if (typeof exports === "object") { // CommonJS @@ -36532,7 +36635,7 @@ module.exports = class zb extends Exchange { return CryptoJS.SHA256; })); -},{"./core":108}],135:[function(require,module,exports){ +},{"./core":117}],144:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -36856,7 +36959,7 @@ module.exports = class zb extends Exchange { return CryptoJS.SHA3; })); -},{"./core":108,"./x64-core":139}],136:[function(require,module,exports){ +},{"./core":117,"./x64-core":148}],145:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -36940,7 +37043,7 @@ module.exports = class zb extends Exchange { return CryptoJS.SHA384; })); -},{"./core":108,"./sha512":137,"./x64-core":139}],137:[function(require,module,exports){ +},{"./core":117,"./sha512":146,"./x64-core":148}],146:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -37264,7 +37367,7 @@ module.exports = class zb extends Exchange { return CryptoJS.SHA512; })); -},{"./core":108,"./x64-core":139}],138:[function(require,module,exports){ +},{"./core":117,"./x64-core":148}],147:[function(require,module,exports){ ;(function (root, factory, undef) { if (typeof exports === "object") { // CommonJS @@ -38035,7 +38138,7 @@ module.exports = class zb extends Exchange { return CryptoJS.TripleDES; })); -},{"./cipher-core":107,"./core":108,"./enc-base64":109,"./evpkdf":111,"./md5":116}],139:[function(require,module,exports){ +},{"./cipher-core":116,"./core":117,"./enc-base64":118,"./evpkdf":120,"./md5":125}],148:[function(require,module,exports){ ;(function (root, factory) { if (typeof exports === "object") { // CommonJS @@ -38340,7 +38443,7 @@ module.exports = class zb extends Exchange { return CryptoJS; })); -},{"./core":108}],140:[function(require,module,exports){ +},{"./core":117}],149:[function(require,module,exports){ (function (self) { 'use strict'; @@ -38841,7 +38944,7 @@ module.exports = class zb extends Exchange { }(typeof self === 'undefined' ? this : self)); -},{}],141:[function(require,module,exports){ +},{}],150:[function(require,module,exports){ // shim for using process in browser var process = module.exports = {}; @@ -39027,7 +39130,7 @@ process.chdir = function (dir) { }; process.umask = function() { return 0; }; -},{}],142:[function(require,module,exports){ +},{}],151:[function(require,module,exports){ 'use strict'; var replace = String.prototype.replace; @@ -39047,7 +39150,7 @@ module.exports = { RFC3986: 'RFC3986' }; -},{}],143:[function(require,module,exports){ +},{}],152:[function(require,module,exports){ 'use strict'; var stringify = require('./stringify'); @@ -39060,7 +39163,7 @@ module.exports = { stringify: stringify }; -},{"./formats":142,"./parse":144,"./stringify":145}],144:[function(require,module,exports){ +},{"./formats":151,"./parse":153,"./stringify":154}],153:[function(require,module,exports){ 'use strict'; var utils = require('./utils'); @@ -39236,7 +39339,7 @@ module.exports = function (str, opts) { return utils.compact(obj); }; -},{"./utils":146}],145:[function(require,module,exports){ +},{"./utils":155}],154:[function(require,module,exports){ 'use strict'; var utils = require('./utils'); @@ -39448,7 +39551,7 @@ module.exports = function (object, opts) { return joined.length > 0 ? prefix + joined : ''; }; -},{"./formats":142,"./utils":146}],146:[function(require,module,exports){ +},{"./formats":151,"./utils":155}],155:[function(require,module,exports){ 'use strict'; var has = Object.prototype.hasOwnProperty; @@ -39653,4 +39756,4 @@ exports.isBuffer = function isBuffer(obj) { }; },{}]},{},[1]) -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","ccxt.browser.js","ccxt.js","js/_1broker.js","js/_1btcxe.js","js/acx.js","js/allcoin.js","js/anxpro.js","js/base/Exchange.js","js/base/Market.js","js/base/errors.js","js/base/functions.js","js/base/throttle.js","js/bibox.js","js/binance.js","js/bit2c.js","js/bitbay.js","js/bitcoincoid.js","js/bitfinex.js","js/bitfinex2.js","js/bitflyer.js","js/bithumb.js","js/bitlish.js","js/bitmarket.js","js/bitmex.js","js/bitso.js","js/bitstamp.js","js/bitstamp1.js","js/bittrex.js","js/bl3p.js","js/bleutrade.js","js/braziliex.js","js/btcbox.js","js/btcchina.js","js/btcexchange.js","js/btcmarkets.js","js/btctradeua.js","js/btcturk.js","js/btcx.js","js/bter.js","js/bxinth.js","js/ccex.js","js/cex.js","js/chbtc.js","js/chilebit.js","js/coincheck.js","js/coinexchange.js","js/coinfloor.js","js/coingi.js","js/coinmarketcap.js","js/coinmate.js","js/coinsecure.js","js/coinspot.js","js/cryptopia.js","js/dsx.js","js/exmo.js","js/flowbtc.js","js/foxbit.js","js/fybse.js","js/fybsg.js","js/gatecoin.js","js/gateio.js","js/gdax.js","js/gemini.js","js/getbtc.js","js/hitbtc.js","js/hitbtc2.js","js/huobi.js","js/huobicny.js","js/huobipro.js","js/independentreserve.js","js/itbit.js","js/jubi.js","js/kraken.js","js/kucoin.js","js/kuna.js","js/lakebtc.js","js/liqui.js","js/livecoin.js","js/luno.js","js/lykke.js","js/mercado.js","js/mixcoins.js","js/nova.js","js/okcoincny.js","js/okcoinusd.js","js/okex.js","js/paymium.js","js/poloniex.js","js/qryptos.js","js/quadrigacx.js","js/quoinex.js","js/southxchange.js","js/surbitcoin.js","js/therock.js","js/tidex.js","js/urdubit.js","js/vaultoro.js","js/vbtc.js","js/virwox.js","js/wex.js","js/xbtce.js","js/yobit.js","js/yunbi.js","js/zaif.js","js/zb.js","node_modules/crypto-js/aes.js","node_modules/crypto-js/cipher-core.js","node_modules/crypto-js/core.js","node_modules/crypto-js/enc-base64.js","node_modules/crypto-js/enc-utf16.js","node_modules/crypto-js/evpkdf.js","node_modules/crypto-js/format-hex.js","node_modules/crypto-js/hmac.js","node_modules/crypto-js/index.js","node_modules/crypto-js/lib-typedarrays.js","node_modules/crypto-js/md5.js","node_modules/crypto-js/mode-cfb.js","node_modules/crypto-js/mode-ctr-gladman.js","node_modules/crypto-js/mode-ctr.js","node_modules/crypto-js/mode-ecb.js","node_modules/crypto-js/mode-ofb.js","node_modules/crypto-js/pad-ansix923.js","node_modules/crypto-js/pad-iso10126.js","node_modules/crypto-js/pad-iso97971.js","node_modules/crypto-js/pad-nopadding.js","node_modules/crypto-js/pad-zeropadding.js","node_modules/crypto-js/pbkdf2.js","node_modules/crypto-js/rabbit-legacy.js","node_modules/crypto-js/rabbit.js","node_modules/crypto-js/rc4.js","node_modules/crypto-js/ripemd160.js","node_modules/crypto-js/sha1.js","node_modules/crypto-js/sha224.js","node_modules/crypto-js/sha256.js","node_modules/crypto-js/sha3.js","node_modules/crypto-js/sha384.js","node_modules/crypto-js/sha512.js","node_modules/crypto-js/tripledes.js","node_modules/crypto-js/x64-core.js","node_modules/fetch-ponyfill/build/fetch-browser.js","node_modules/process/browser.js","node_modules/qs/lib/formats.js","node_modules/qs/lib/index.js","node_modules/qs/lib/parse.js","node_modules/qs/lib/stringify.js","node_modules/qs/lib/utils.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;;ACLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7PA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9YA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACnOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;ACp4BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/XA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChhBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACn1BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3sBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1VA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChWA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnXA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/PA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3MA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACndA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnYA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/SA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1QA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzdA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtUA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACznBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5SA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/ZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/iBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACx2BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxjCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/PA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpeA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/PA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtyBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/qBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtjBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnXA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrSA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/oBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7yBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5PA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3RA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtWA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/2BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvvBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3QA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1QA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClUA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClUA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjwBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/SA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACXA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","/*  A entry point for the browser bundle version. This gets compiled by:\n        \n        browserify --debug ./ccxt.browser.js > ./build/ccxt.browser.js\n */\n\nwindow.ccxt = require ('./ccxt')","\"use strict\";\n\n/*\n\nMIT License\n\nCopyright (c) 2017 Igor Kroitor\n\nPermission is hereby granted, free of charge, to any person obtaining a copy\nof this software and associated documentation files (the \"Software\"), to deal\nin the Software without restriction, including without limitation the rights\nto use, copy, modify, merge, publish, distribute, sublicense, and/or sell\ncopies of the Software, and to permit persons to whom the Software is\nfurnished to do so, subject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\nFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\nAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\nLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\nOUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\nSOFTWARE.\n\n*/\n\n\"use strict\";\n\n//-----------------------------------------------------------------------------\n\nconst Exchange  = require ('./js/base/Exchange')\nconst functions = require ('./js/base/functions')\nconst errors    = require ('./js/base/errors')\n\n//-----------------------------------------------------------------------------\n// this is updated by vss.js when building\n\nconst version = '1.10.760'\n\nExchange.ccxtVersion = version\n\n//-----------------------------------------------------------------------------\n\nconst exchanges = {\n    '_1broker':                require ('./js/_1broker.js'),\n    '_1btcxe':                 require ('./js/_1btcxe.js'),\n    'acx':                     require ('./js/acx.js'),\n    'allcoin':                 require ('./js/allcoin.js'),\n    'anxpro':                  require ('./js/anxpro.js'),\n    'bibox':                   require ('./js/bibox.js'),\n    'binance':                 require ('./js/binance.js'),\n    'bit2c':                   require ('./js/bit2c.js'),\n    'bitbay':                  require ('./js/bitbay.js'),\n    'bitcoincoid':             require ('./js/bitcoincoid.js'),\n    'bitfinex':                require ('./js/bitfinex.js'),\n    'bitfinex2':               require ('./js/bitfinex2.js'),\n    'bitflyer':                require ('./js/bitflyer.js'),\n    'bithumb':                 require ('./js/bithumb.js'),\n    'bitlish':                 require ('./js/bitlish.js'),\n    'bitmarket':               require ('./js/bitmarket.js'),\n    'bitmex':                  require ('./js/bitmex.js'),\n    'bitso':                   require ('./js/bitso.js'),\n    'bitstamp':                require ('./js/bitstamp.js'),\n    'bitstamp1':               require ('./js/bitstamp1.js'),\n    'bittrex':                 require ('./js/bittrex.js'),\n    'bl3p':                    require ('./js/bl3p.js'),\n    'bleutrade':               require ('./js/bleutrade.js'),\n    'braziliex':               require ('./js/braziliex.js'),\n    'btcbox':                  require ('./js/btcbox.js'),\n    'btcchina':                require ('./js/btcchina.js'),\n    'btcexchange':             require ('./js/btcexchange.js'),\n    'btcmarkets':              require ('./js/btcmarkets.js'),\n    'btctradeua':              require ('./js/btctradeua.js'),\n    'btcturk':                 require ('./js/btcturk.js'),\n    'btcx':                    require ('./js/btcx.js'),\n    'bter':                    require ('./js/bter.js'),\n    'bxinth':                  require ('./js/bxinth.js'),\n    'ccex':                    require ('./js/ccex.js'),\n    'cex':                     require ('./js/cex.js'),\n    'chbtc':                   require ('./js/chbtc.js'),\n    'chilebit':                require ('./js/chilebit.js'),\n    'coincheck':               require ('./js/coincheck.js'),\n    'coinexchange':            require ('./js/coinexchange.js'),\n    'coinfloor':               require ('./js/coinfloor.js'),\n    'coingi':                  require ('./js/coingi.js'),\n    'coinmarketcap':           require ('./js/coinmarketcap.js'),\n    'coinmate':                require ('./js/coinmate.js'),\n    'coinsecure':              require ('./js/coinsecure.js'),\n    'coinspot':                require ('./js/coinspot.js'),\n    'cryptopia':               require ('./js/cryptopia.js'),\n    'dsx':                     require ('./js/dsx.js'),\n    'exmo':                    require ('./js/exmo.js'),\n    'flowbtc':                 require ('./js/flowbtc.js'),\n    'foxbit':                  require ('./js/foxbit.js'),\n    'fybse':                   require ('./js/fybse.js'),\n    'fybsg':                   require ('./js/fybsg.js'),\n    'gatecoin':                require ('./js/gatecoin.js'),\n    'gateio':                  require ('./js/gateio.js'),\n    'gdax':                    require ('./js/gdax.js'),\n    'gemini':                  require ('./js/gemini.js'),\n    'getbtc':                  require ('./js/getbtc.js'),\n    'hitbtc':                  require ('./js/hitbtc.js'),\n    'hitbtc2':                 require ('./js/hitbtc2.js'),\n    'huobi':                   require ('./js/huobi.js'),\n    'huobicny':                require ('./js/huobicny.js'),\n    'huobipro':                require ('./js/huobipro.js'),\n    'independentreserve':      require ('./js/independentreserve.js'),\n    'itbit':                   require ('./js/itbit.js'),\n    'jubi':                    require ('./js/jubi.js'),\n    'kraken':                  require ('./js/kraken.js'),\n    'kucoin':                  require ('./js/kucoin.js'),\n    'kuna':                    require ('./js/kuna.js'),\n    'lakebtc':                 require ('./js/lakebtc.js'),\n    'liqui':                   require ('./js/liqui.js'),\n    'livecoin':                require ('./js/livecoin.js'),\n    'luno':                    require ('./js/luno.js'),\n    'lykke':                   require ('./js/lykke.js'),\n    'mercado':                 require ('./js/mercado.js'),\n    'mixcoins':                require ('./js/mixcoins.js'),\n    'nova':                    require ('./js/nova.js'),\n    'okcoincny':               require ('./js/okcoincny.js'),\n    'okcoinusd':               require ('./js/okcoinusd.js'),\n    'okex':                    require ('./js/okex.js'),\n    'paymium':                 require ('./js/paymium.js'),\n    'poloniex':                require ('./js/poloniex.js'),\n    'qryptos':                 require ('./js/qryptos.js'),\n    'quadrigacx':              require ('./js/quadrigacx.js'),\n    'quoinex':                 require ('./js/quoinex.js'),\n    'southxchange':            require ('./js/southxchange.js'),\n    'surbitcoin':              require ('./js/surbitcoin.js'),\n    'therock':                 require ('./js/therock.js'),\n    'tidex':                   require ('./js/tidex.js'),\n    'urdubit':                 require ('./js/urdubit.js'),\n    'vaultoro':                require ('./js/vaultoro.js'),\n    'vbtc':                    require ('./js/vbtc.js'),\n    'virwox':                  require ('./js/virwox.js'),\n    'wex':                     require ('./js/wex.js'),\n    'xbtce':                   require ('./js/xbtce.js'),\n    'yobit':                   require ('./js/yobit.js'),\n    'yunbi':                   require ('./js/yunbi.js'),\n    'zaif':                    require ('./js/zaif.js'),\n    'zb':                      require ('./js/zb.js'),    \n}\n\n//-----------------------------------------------------------------------------\n\nmodule.exports = Object.assign ({ version, Exchange, exchanges: Object.keys (exchanges) }, exchanges, functions, errors)\n\n//-----------------------------------------------------------------------------\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class _1broker extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': '_1broker',\n            'name': '1Broker',\n            'countries': 'US',\n            'rateLimit': 1500,\n            'version': 'v2',\n            'hasPublicAPI': false,\n            'hasCORS': true,\n            'hasFetchTrades': false,\n            'hasFetchOHLCV': true,\n            'timeframes': {\n                '1m': '60',\n                '15m': '900',\n                '1h': '3600',\n                '1d': '86400',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766021-420bd9fc-5ecb-11e7-8ed6-56d0081efed2.jpg',\n                'api': 'https://1broker.com/api',\n                'www': 'https://1broker.com',\n                'doc': 'https://1broker.com/?c=en/content/api-documentation',\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': false,\n            },\n            'api': {\n                'private': {\n                    'get': [\n                        'market/bars',\n                        'market/categories',\n                        'market/details',\n                        'market/list',\n                        'market/quotes',\n                        'market/ticks',\n                        'order/cancel',\n                        'order/create',\n                        'order/open',\n                        'position/close',\n                        'position/close_cancel',\n                        'position/edit',\n                        'position/history',\n                        'position/open',\n                        'position/shared/get',\n                        'social/profile_statistics',\n                        'social/profile_trades',\n                        'user/bitcoin_deposit_address',\n                        'user/details',\n                        'user/overview',\n                        'user/quota_status',\n                        'user/transaction_log',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchCategories () {\n        let response = await this.privateGetMarketCategories ();\n        // they return an empty string among their categories, wtf?\n        let categories = response['response'];\n        let result = [];\n        for (let i = 0; i < categories.length; i++) {\n            if (categories[i])\n                result.push (categories[i]);\n        }\n        return result;\n    }\n\n    async fetchMarkets () {\n        let this_ = this; // workaround for Babel bug (not passing `this` to _recursive() call)\n        let categories = await this.fetchCategories ();\n        let result = [];\n        for (let c = 0; c < categories.length; c++) {\n            let category = categories[c];\n            let markets = await this_.privateGetMarketList ({\n                'category': category.toLowerCase (),\n            });\n            for (let p = 0; p < markets['response'].length; p++) {\n                let market = markets['response'][p];\n                let id = market['symbol'];\n                let symbol = undefined;\n                let base = undefined;\n                let quote = undefined;\n                if ((category == 'FOREX') || (category == 'CRYPTO')) {\n                    symbol = market['name'];\n                    let parts = symbol.split ('/');\n                    base = parts[0];\n                    quote = parts[1];\n                } else {\n                    base = id;\n                    quote = 'USD';\n                    symbol = base + '/' + quote;\n                }\n                base = this_.commonCurrencyCode (base);\n                quote = this_.commonCurrencyCode (quote);\n                result.push ({\n                    'id': id,\n                    'symbol': symbol,\n                    'base': base,\n                    'quote': quote,\n                    'info': market,\n                });\n            }\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balance = await this.privateGetUserOverview ();\n        let response = balance['response'];\n        let result = {\n            'info': response,\n        };\n        let currencies = Object.keys (this.currencies);\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            result[currency] = this.account ();\n        }\n        let total = parseFloat (response['balance']);\n        result['BTC']['free'] = total;\n        result['BTC']['total'] = total;\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetMarketQuotes (this.extend ({\n            'symbols': this.marketId (symbol),\n        }, params));\n        let orderbook = response['response'][0];\n        let timestamp = this.parse8601 (orderbook['updated']);\n        let bidPrice = parseFloat (orderbook['bid']);\n        let askPrice = parseFloat (orderbook['ask']);\n        let bid = [ bidPrice, undefined ];\n        let ask = [ askPrice, undefined ];\n        return {\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'bids': [ bid ],\n            'asks': [ ask ],\n        };\n    }\n\n    async fetchTrades (symbol) {\n        throw new ExchangeError (this.id + ' fetchTrades () method not implemented yet');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let result = await this.privateGetMarketBars (this.extend ({\n            'symbol': this.marketId (symbol),\n            'resolution': 60,\n            'limit': 1,\n        }, params));\n        let orderbook = await this.fetchOrderBook (symbol);\n        let ticker = result['response'][0];\n        let timestamp = this.parse8601 (ticker['date']);\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['h']),\n            'low': parseFloat (ticker['l']),\n            'bid': orderbook['bids'][0][0],\n            'ask': orderbook['asks'][0][0],\n            'vwap': undefined,\n            'open': parseFloat (ticker['o']),\n            'close': parseFloat (ticker['c']),\n            'first': undefined,\n            'last': undefined,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            this.parse8601 (ohlcv['date']),\n            parseFloat (ohlcv['o']),\n            parseFloat (ohlcv['h']),\n            parseFloat (ohlcv['l']),\n            parseFloat (ohlcv['c']),\n            undefined,\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n            'resolution': this.timeframes[timeframe],\n        };\n        if (since)\n            request['date_start'] = this.iso8601 (since); // they also support date_end\n        if (limit)\n            request['limit'] = limit;\n        let result = await this.privateGetMarketBars (this.extend (request, params));\n        return this.parseOHLCVs (result['response'], market, timeframe, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'symbol': this.marketId (symbol),\n            'margin': amount,\n            'direction': (side == 'sell') ? 'short' : 'long',\n            'leverage': 1,\n            'type': side,\n        };\n        if (type == 'limit')\n            order['price'] = price;\n        else\n            order['type'] += '_market';\n        let result = await this.privateGetOrderCreate (this.extend (order, params));\n        return {\n            'info': result,\n            'id': result['response']['order_id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostOrderCancel ({ 'order_id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        this.checkRequiredCredentials ();\n        let url = this.urls['api'] + '/' + this.version + '/' + path + '.php';\n        let query = this.extend ({ 'token': this.apiKey }, params);\n        url += '?' + this.urlencode (query);\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('warning' in response)\n            if (response['warning'])\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        if ('error' in response)\n            if (response['error'])\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class _1btcxe extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': '_1btcxe',\n            'name': '1BTCXE',\n            'countries': 'PA', // Panama\n            'comment': 'Crypto Capital API',\n            'hasCORS': true,\n            'hasFetchOHLCV': true,\n            'hasWithdraw': true,\n            'timeframes': {\n                '1d': '1year',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766049-2b294408-5ecc-11e7-85cc-adaff013dc1a.jpg',\n                'api': 'https://1btcxe.com/api',\n                'www': 'https://1btcxe.com',\n                'doc': 'https://1btcxe.com/api-docs.php',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'stats',\n                        'historical-prices',\n                        'order-book',\n                        'transactions',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balances-and-info',\n                        'open-orders',\n                        'user-transactions',\n                        'btc-deposit-address/get',\n                        'btc-deposit-address/new',\n                        'deposits/get',\n                        'withdrawals/get',\n                        'orders/new',\n                        'orders/edit',\n                        'orders/cancel',\n                        'orders/status',\n                        'withdrawals/new',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/USD': { 'id': 'USD', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD' },\n                'BTC/EUR': { 'id': 'EUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR' },\n                'BTC/CNY': { 'id': 'CNY', 'symbol': 'BTC/CNY', 'base': 'BTC', 'quote': 'CNY' },\n                'BTC/RUB': { 'id': 'RUB', 'symbol': 'BTC/RUB', 'base': 'BTC', 'quote': 'RUB' },\n                'BTC/CHF': { 'id': 'CHF', 'symbol': 'BTC/CHF', 'base': 'BTC', 'quote': 'CHF' },\n                'BTC/JPY': { 'id': 'JPY', 'symbol': 'BTC/JPY', 'base': 'BTC', 'quote': 'JPY' },\n                'BTC/GBP': { 'id': 'GBP', 'symbol': 'BTC/GBP', 'base': 'BTC', 'quote': 'GBP' },\n                'BTC/CAD': { 'id': 'CAD', 'symbol': 'BTC/CAD', 'base': 'BTC', 'quote': 'CAD' },\n                'BTC/AUD': { 'id': 'AUD', 'symbol': 'BTC/AUD', 'base': 'BTC', 'quote': 'AUD' },\n                'BTC/AED': { 'id': 'AED', 'symbol': 'BTC/AED', 'base': 'BTC', 'quote': 'AED' },\n                'BTC/BGN': { 'id': 'BGN', 'symbol': 'BTC/BGN', 'base': 'BTC', 'quote': 'BGN' },\n                'BTC/CZK': { 'id': 'CZK', 'symbol': 'BTC/CZK', 'base': 'BTC', 'quote': 'CZK' },\n                'BTC/DKK': { 'id': 'DKK', 'symbol': 'BTC/DKK', 'base': 'BTC', 'quote': 'DKK' },\n                'BTC/HKD': { 'id': 'HKD', 'symbol': 'BTC/HKD', 'base': 'BTC', 'quote': 'HKD' },\n                'BTC/HRK': { 'id': 'HRK', 'symbol': 'BTC/HRK', 'base': 'BTC', 'quote': 'HRK' },\n                'BTC/HUF': { 'id': 'HUF', 'symbol': 'BTC/HUF', 'base': 'BTC', 'quote': 'HUF' },\n                'BTC/ILS': { 'id': 'ILS', 'symbol': 'BTC/ILS', 'base': 'BTC', 'quote': 'ILS' },\n                'BTC/INR': { 'id': 'INR', 'symbol': 'BTC/INR', 'base': 'BTC', 'quote': 'INR' },\n                'BTC/MUR': { 'id': 'MUR', 'symbol': 'BTC/MUR', 'base': 'BTC', 'quote': 'MUR' },\n                'BTC/MXN': { 'id': 'MXN', 'symbol': 'BTC/MXN', 'base': 'BTC', 'quote': 'MXN' },\n                'BTC/NOK': { 'id': 'NOK', 'symbol': 'BTC/NOK', 'base': 'BTC', 'quote': 'NOK' },\n                'BTC/NZD': { 'id': 'NZD', 'symbol': 'BTC/NZD', 'base': 'BTC', 'quote': 'NZD' },\n                'BTC/PLN': { 'id': 'PLN', 'symbol': 'BTC/PLN', 'base': 'BTC', 'quote': 'PLN' },\n                'BTC/RON': { 'id': 'RON', 'symbol': 'BTC/RON', 'base': 'BTC', 'quote': 'RON' },\n                'BTC/SEK': { 'id': 'SEK', 'symbol': 'BTC/SEK', 'base': 'BTC', 'quote': 'SEK' },\n                'BTC/SGD': { 'id': 'SGD', 'symbol': 'BTC/SGD', 'base': 'BTC', 'quote': 'SGD' },\n                'BTC/THB': { 'id': 'THB', 'symbol': 'BTC/THB', 'base': 'BTC', 'quote': 'THB' },\n                'BTC/TRY': { 'id': 'TRY', 'symbol': 'BTC/TRY', 'base': 'BTC', 'quote': 'TRY' },\n                'BTC/ZAR': { 'id': 'ZAR', 'symbol': 'BTC/ZAR', 'base': 'BTC', 'quote': 'ZAR' },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostBalancesAndInfo ();\n        let balance = response['balances-and-info'];\n        let result = { 'info': balance };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            account['free'] = this.safeFloat (balance['available'], currency, 0.0);\n            account['used'] = this.safeFloat (balance['on_hold'], currency, 0.0);\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let response = await this.publicGetOrderBook (this.extend ({\n            'currency': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (response['order-book'], undefined, 'bid', 'ask', 'price', 'order_amount');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let response = await this.publicGetStats (this.extend ({\n            'currency': this.marketId (symbol),\n        }, params));\n        let ticker = response['stats'];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['max']),\n            'low': parseFloat (ticker['min']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': parseFloat (ticker['open']),\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last_price']),\n            'change': parseFloat (ticker['daily_change']),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': parseFloat (ticker['total_btc_traded']),\n        };\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1d', since = undefined, limit = undefined) {\n        return [\n            this.parse8601 (ohlcv['date'] + ' 00:00:00'),\n            undefined,\n            undefined,\n            undefined,\n            parseFloat (ohlcv['price']),\n            undefined,\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1d', since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetHistoricalPrices (this.extend ({\n            'currency': market['id'],\n            'timeframe': this.timeframes[timeframe],\n        }, params));\n        let ohlcvs = this.omit (response['historical-prices'], 'request_currency');\n        return this.parseOHLCVs (ohlcvs, market, timeframe, since, limit);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['timestamp']) * 1000;\n        return {\n            'id': trade['id'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': undefined,\n            'type': undefined,\n            'side': trade['maker_type'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetTransactions (this.extend ({\n            'currency': market['id'],\n        }, params));\n        let trades = this.omit (response['transactions'], 'request_currency');\n        return this.parseTrades (trades, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let order = {\n            'side': side,\n            'type': type,\n            'currency': this.marketId (symbol),\n            'amount': amount,\n        };\n        if (type === 'limit')\n            order['limit_price'] = price;\n        let result = await this.privatePostOrdersNew (this.extend (order, params));\n        return {\n            'info': result,\n            'id': result,\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostOrdersCancel ({ 'id': id });\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostWithdrawalsNew (this.extend ({\n            'currency': currency,\n            'amount': parseFloat (amount),\n            'address': address,\n        }, params));\n        return {\n            'info': response,\n            'id': response['result']['uuid'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        if (this.id === 'cryptocapital')\n            throw new ExchangeError (this.id + ' is an abstract base API for _1btcxe');\n        let url = this.urls['api'] + '/' + path;\n        if (api === 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let query = this.extend ({\n                'api_key': this.apiKey,\n                'nonce': this.nonce (),\n            }, params);\n            let request = this.json (query);\n            query['signature'] = this.hmac (this.encode (request), this.encode (this.secret));\n            body = this.json (query);\n            headers = { 'Content-Type': 'application/json' };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('errors' in response) {\n            let errors = [];\n            for (let e = 0; e < response['errors'].length; e++) {\n                let error = response['errors'][e];\n                errors.push (error['code'] + ': ' + error['message']);\n            }\n            errors = errors.join (' ');\n            throw new ExchangeError (this.id + ' ' + errors);\n        }\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, OrderNotFound } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class acx extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'acx',\n            'name': 'ACX',\n            'countries': 'AU',\n            'rateLimit': 1000,\n            'version': 'v2',\n            'hasCORS': true,\n            'hasFetchTickers': true,\n            'hasFetchOHLCV': true,\n            'hasWithdraw': true,\n            'timeframes': {\n                '1m': '1',\n                '5m': '5',\n                '15m': '15',\n                '30m': '30',\n                '1h': '60',\n                '2h': '120',\n                '4h': '240',\n                '12h': '720',\n                '1d': '1440',\n                '3d': '4320',\n                '1w': '10080',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30247614-1fe61c74-9621-11e7-9e8c-f1a627afa279.jpg',\n                'extension': '.json',\n                'api': 'https://acx.io/api',\n                'www': 'https://acx.io',\n                'doc': 'https://acx.io/documents/api_v2',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'markets', // Get all available markets\n                        'tickers', // Get ticker of all markets\n                        'tickers/{market}', // Get ticker of specific market\n                        'trades', // Get recent trades on market, each trade is included only once Trades are sorted in reverse creation order.\n                        'order_book', // Get the order book of specified market\n                        'depth', // Get depth or specified market Both asks and bids are sorted from highest price to lowest.\n                        'k', // Get OHLC(k line) of specific market\n                        'k_with_pending_trades', // Get K data with pending trades, which are the trades not included in K data yet, because there's delay between trade generated and processed by K data generator\n                        'timestamp', // Get server current time, in seconds since Unix epoch\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'members/me', // Get your profile and accounts info\n                        'deposits', // Get your deposits history\n                        'deposit', // Get details of specific deposit\n                        'deposit_address', // Where to deposit The address field could be empty when a new address is generating (e.g. for bitcoin), you should try again later in that case.\n                        'orders', // Get your orders, results is paginated\n                        'order', // Get information of specified order\n                        'trades/my', // Get your executed trades Trades are sorted in reverse creation order.\n                        'withdraws', // Get your cryptocurrency withdraws\n                        'withdraw', // Get your cryptocurrency withdraw\n                    ],\n                    'post': [\n                        'orders', // Create a Sell/Buy order\n                        'orders/multi', // Create multiple sell/buy orders\n                        'orders/clear', // Cancel all my orders\n                        'order/delete', // Cancel an order\n                        'withdraw', // Create a withdraw\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'maker': 0.0,\n                    'taker': 0.0,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'withdraw': 0.0, // There is only 1% fee on withdrawals to your bank account.\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetMarkets ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['id'];\n            let symbol = market['name'];\n            let [ base, quote ] = symbol.split ('/');\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetMembersMe ();\n        let balances = response['accounts'];\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let uppercase = currency.toUpperCase ();\n            let account = {\n                'free': parseFloat (balance['balance']),\n                'used': parseFloat (balance['locked']),\n                'total': 0.0,\n            };\n            account['total'] = this.sum (account['free'], account['used']);\n            result[uppercase] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orderbook = await this.publicGetDepth (this.extend ({\n            'market': market['id'],\n            'limit': 300,\n        }, params));\n        let timestamp = orderbook['timestamp'] * 1000;\n        let result = this.parseOrderBook (orderbook, timestamp);\n        result['bids'] = this.sortBy (result['bids'], 0, true);\n        result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['at'] * 1000;\n        ticker = ticker['ticker'];\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high', undefined),\n            'low': this.safeFloat (ticker, 'low', undefined),\n            'bid': this.safeFloat (ticker, 'buy', undefined),\n            'ask': this.safeFloat (ticker, 'sell', undefined),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last', undefined),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'vol', undefined),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTickers (params);\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = undefined;\n            let symbol = id;\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            } else {\n                let base = id.slice (0, 3);\n                let quote = id.slice (3, 6);\n                base = base.toUpperCase ();\n                quote = quote.toUpperCase ();\n                base = this.commonCurrencyCode (base);\n                quote = this.commonCurrencyCode (quote);\n                symbol = base + '/' + quote;\n            }\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTickersMarket (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseTicker (response, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = this.parse8601 (trade['created_at']);\n        return {\n            'id': trade['id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': this.safeFloat (trade, 'price'),\n            'amount': this.safeFloat (trade, 'volume'),\n            'cost': this.safeFloat (trade, 'funds'),\n            'info': trade,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTrades (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv[0] * 1000,\n            ohlcv[1],\n            ohlcv[2],\n            ohlcv[3],\n            ohlcv[4],\n            ohlcv[5],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        if (!limit)\n            limit = 500; // default is 30\n        let request = {\n            'market': market['id'],\n            'period': this.timeframes[timeframe],\n            'limit': limit,\n        };\n        if (since)\n            request['timestamp'] = since;\n        let response = await this.publicGetK (this.extend (request, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    parseOrder (order, market = undefined) {\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else {\n            let marketId = order['market'];\n            symbol = this.marketsById[marketId]['symbol'];\n        }\n        let timestamp = this.parse8601 (order['created_at']);\n        let state = order['state'];\n        let status = undefined;\n        if (state === 'done') {\n            status = 'closed';\n        } else if (state === 'wait') {\n            status = 'open';\n        } else if (state === 'cancel') {\n            status = 'canceled';\n        }\n        return {\n            'id': order['id'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': status,\n            'symbol': symbol,\n            'type': order['ord_type'],\n            'side': order['side'],\n            'price': parseFloat (order['price']),\n            'amount': parseFloat (order['volume']),\n            'filled': parseFloat (order['executed_volume']),\n            'remaining': parseFloat (order['remaining_volume']),\n            'trades': undefined,\n            'fee': undefined,\n            'info': order,\n        };\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'market': this.marketId (symbol),\n            'side': side,\n            'volume': amount.toString (),\n            'ord_type': type,\n        };\n        if (type === 'limit') {\n            order['price'] = price.toString ();\n        }\n        let response = await this.privatePostOrders (this.extend (order, params));\n        let market = this.marketsById[response['market']];\n        return this.parseOrder (response, market);\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let result = await this.privatePostOrderDelete ({ 'id': id });\n        let order = this.parseOrder (result);\n        if (order['status'] === 'closed') {\n            throw new OrderNotFound (this.id + ' ' + result);\n        }\n        return order;\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let result = await this.privatePostWithdraw (this.extend ({\n            'currency': currency.toLowerCase (),\n            'sum': amount,\n            'address': address,\n        }, params));\n        return {\n            'info': result,\n            'id': undefined,\n        };\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    encodeParams (params) {\n        if ('orders' in params) {\n            let orders = params['orders'];\n            let query = this.urlencode (this.keysort (this.omit (params, 'orders')));\n            for (let i = 0; i < orders.length; i++) {\n                let order = orders[i];\n                let keys = Object.keys (order);\n                for (let k = 0; k < keys.length; k++) {\n                    let key = keys[k];\n                    let value = order[key];\n                    query += '&orders%5B%5D%5B' + key + '%5D=' + value.toString ();\n                }\n            }\n            return query;\n        }\n        return this.urlencode (this.keysort (params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let request = '/api' + '/' + this.version + '/' + this.implodeParams (path, params);\n        if ('extension' in this.urls)\n            request += this.urls['extension'];\n        let query = this.omit (params, this.extractParams (path));\n        let url = this.urls['api'] + request;\n        if (api === 'public') {\n            if (Object.keys (query).length) {\n                url += '?' + this.urlencode (query);\n            }\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let query = this.encodeParams (this.extend ({\n                'access_key': this.apiKey,\n                'tonce': nonce,\n            }, params));\n            let auth = method + '|' + request + '|' + query;\n            let signature = this.hmac (this.encode (auth), this.encode (this.secret));\n            let suffix = query + '&signature=' + signature;\n            if (method === 'GET') {\n                url += '?' + suffix;\n            } else {\n                body = suffix;\n                headers = { 'Content-Type': 'application/x-www-form-urlencoded' };\n            }\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst okcoinusd = require ('./okcoinusd.js');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class allcoin extends okcoinusd {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'allcoin',\n            'name': 'Allcoin',\n            'countries': 'CA',\n            'hasCORS': false,\n            'extension': '',\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/31561809-c316b37c-b061-11e7-8d5a-b547b4d730eb.jpg',\n                'api': {\n                    'web': 'https://www.allcoin.com',\n                    'public': 'https://api.allcoin.com/api',\n                    'private': 'https://api.allcoin.com/api',\n                },\n                'www': 'https://www.allcoin.com',\n                'doc': 'https://www.allcoin.com/About/APIReference',\n            },\n            'api': {\n                'web': {\n                    'get': [\n                        'Home/MarketOverViewDetail/',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'depth',\n                        'kline',\n                        'ticker',\n                        'trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'batch_trade',\n                        'cancel_order',\n                        'order_history',\n                        'order_info',\n                        'orders_info',\n                        'repayment',\n                        'trade',\n                        'trade_history',\n                        'userinfo',\n                    ],\n                },\n            },\n            'markets': undefined,\n        });\n    }\n\n    async fetchMarkets () {\n        let result = [];\n        let response = await this.webGetHomeMarketOverViewDetail ();\n        let coins = response['marketCoins'];\n        for (let j = 0; j < coins.length; j++) {\n            let markets = coins[j]['Markets'];\n            for (let k = 0; k < markets.length; k++) {\n                let market = markets[k]['Market'];\n                let base = market['Primary'];\n                let quote = market['Secondary'];\n                let id = base.toLowerCase () + '_' + quote.toLowerCase ();\n                let symbol = base + '/' + quote;\n                result.push ({\n                    'id': id,\n                    'symbol': symbol,\n                    'base': base,\n                    'quote': quote,\n                    'type': 'spot',\n                    'spot': true,\n                    'future': false,\n                    'info': market,\n                });\n            }\n        }\n        return result;\n    }\n\n    parseOrderStatus (status) {\n        if (status === -1)\n            return 'canceled';\n        if (status === 0)\n            return 'open';\n        if (status === 1)\n            return 'open'; // partially filled\n        if (status === 2)\n            return 'closed';\n        if (status === 10)\n            return 'canceled';\n        return status;\n    }\n\n    getCreateDateField () {\n        // allcoin typo create_data instead of create_date\n        return 'create_data';\n    }\n\n    getOrdersField () {\n        // allcoin typo order instead of orders (expected based on their API docs)\n        return 'order';\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class anxpro extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'anxpro',\n            'name': 'ANXPro',\n            'countries': [ 'JP', 'SG', 'HK', 'NZ' ],\n            'version': '2',\n            'rateLimit': 1500,\n            'hasCORS': false,\n            'hasFetchTrades': false,\n            'hasWithdraw': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27765983-fd8595da-5ec9-11e7-82e3-adb3ab8c2612.jpg',\n                'api': 'https://anxpro.com/api',\n                'www': 'https://anxpro.com',\n                'doc': [\n                    'http://docs.anxv2.apiary.io',\n                    'https://anxpro.com/pages/api',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        '{currency_pair}/money/ticker',\n                        '{currency_pair}/money/depth/full',\n                        '{currency_pair}/money/trade/fetch', // disabled by ANXPro\n                    ],\n                },\n                'private': {\n                    'post': [\n                        '{currency_pair}/money/order/add',\n                        '{currency_pair}/money/order/cancel',\n                        '{currency_pair}/money/order/quote',\n                        '{currency_pair}/money/order/result',\n                        '{currency_pair}/money/orders',\n                        'money/{currency}/address',\n                        'money/{currency}/send_simple',\n                        'money/info',\n                        'money/trade/list',\n                        'money/wallet/history',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/USD': { 'id': 'BTCUSD', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD', 'multiplier': 100000 },\n                'BTC/HKD': { 'id': 'BTCHKD', 'symbol': 'BTC/HKD', 'base': 'BTC', 'quote': 'HKD', 'multiplier': 100000 },\n                'BTC/EUR': { 'id': 'BTCEUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR', 'multiplier': 100000 },\n                'BTC/CAD': { 'id': 'BTCCAD', 'symbol': 'BTC/CAD', 'base': 'BTC', 'quote': 'CAD', 'multiplier': 100000 },\n                'BTC/AUD': { 'id': 'BTCAUD', 'symbol': 'BTC/AUD', 'base': 'BTC', 'quote': 'AUD', 'multiplier': 100000 },\n                'BTC/SGD': { 'id': 'BTCSGD', 'symbol': 'BTC/SGD', 'base': 'BTC', 'quote': 'SGD', 'multiplier': 100000 },\n                'BTC/JPY': { 'id': 'BTCJPY', 'symbol': 'BTC/JPY', 'base': 'BTC', 'quote': 'JPY', 'multiplier': 100000 },\n                'BTC/GBP': { 'id': 'BTCGBP', 'symbol': 'BTC/GBP', 'base': 'BTC', 'quote': 'GBP', 'multiplier': 100000 },\n                'BTC/NZD': { 'id': 'BTCNZD', 'symbol': 'BTC/NZD', 'base': 'BTC', 'quote': 'NZD', 'multiplier': 100000 },\n                'LTC/BTC': { 'id': 'LTCBTC', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC', 'multiplier': 100000 },\n                'STR/BTC': { 'id': 'STRBTC', 'symbol': 'STR/BTC', 'base': 'STR', 'quote': 'BTC', 'multiplier': 100000000 },\n                'XRP/BTC': { 'id': 'XRPBTC', 'symbol': 'XRP/BTC', 'base': 'XRP', 'quote': 'BTC', 'multiplier': 100000000 },\n                'DOGE/BTC': { 'id': 'DOGEBTC', 'symbol': 'DOGE/BTC', 'base': 'DOGE', 'quote': 'BTC', 'multiplier': 100000000 },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.3 / 100,\n                    'taker': 0.6 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostMoneyInfo ();\n        let balance = response['data'];\n        let currencies = Object.keys (balance['Wallets']);\n        let result = { 'info': balance };\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let account = this.account ();\n            if (currency in balance['Wallets']) {\n                let wallet = balance['Wallets'][currency];\n                account['free'] = parseFloat (wallet['Available_Balance']['value']);\n                account['total'] = parseFloat (wallet['Balance']['value']);\n                account['used'] = account['total'] - account['free'];\n            }\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let response = await this.publicGetCurrencyPairMoneyDepthFull (this.extend ({\n            'currency_pair': this.marketId (symbol),\n        }, params));\n        let orderbook = response['data'];\n        let t = parseInt (orderbook['dataUpdateTime']);\n        let timestamp = parseInt (t / 1000);\n        return this.parseOrderBook (orderbook, timestamp, 'bids', 'asks', 'price', 'amount');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let response = await this.publicGetCurrencyPairMoneyTicker (this.extend ({\n            'currency_pair': this.marketId (symbol),\n        }, params));\n        let ticker = response['data'];\n        let t = parseInt (ticker['dataUpdateTime']);\n        let timestamp = parseInt (t / 1000);\n        let bid = this.safeFloat (ticker['buy'], 'value');\n        let ask = this.safeFloat (ticker['sell'], 'value');\n        let baseVolume = parseFloat (ticker['vol']['value']);\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']['value']),\n            'low': parseFloat (ticker['low']['value']),\n            'bid': bid,\n            'ask': ask,\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']['value']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': parseFloat (ticker['avg']['value']),\n            'baseVolume': baseVolume,\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        throw new ExchangeError (this.id + ' switched off the trades endpoint, see their docs at http://docs.anxv2.apiary.io/reference/market-data/currencypairmoneytradefetch-disabled');\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let market = this.market (symbol);\n        let order = {\n            'currency_pair': market['id'],\n            'amount_int': parseInt (amount * 100000000), // 10^8\n        };\n        if (type === 'limit') {\n            order['price_int'] = parseInt (price * market['multiplier']); // 10^5 or 10^8\n        }\n        order['type'] = (side === 'buy') ? 'bid' : 'ask';\n        let result = await this.privatePostCurrencyPairMoneyOrderAdd (this.extend (order, params));\n        return {\n            'info': result,\n            'id': result['data'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCurrencyPairMoneyOrderCancel ({ 'oid': id });\n    }\n\n    getAmountMultiplier (currency) {\n        if (currency === 'BTC') {\n            return 100000000;\n        } else if (currency === 'LTC') {\n            return 100000000;\n        } else if (currency === 'STR') {\n            return 100000000;\n        } else if (currency === 'XRP') {\n            return 100000000;\n        } else if (currency === 'DOGE') {\n            return 100000000;\n        }\n        return 100;\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let multiplier = this.getAmountMultiplier (currency);\n        let response = await this.privatePostMoneyCurrencySendSimple (this.extend ({\n            'currency': currency,\n            'amount_int': parseInt (amount * multiplier),\n            'address': address,\n        }, params));\n        return {\n            'info': response,\n            'id': response['data']['transactionId'],\n        };\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let request = this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        let url = this.urls['api'] + '/' + this.version + '/' + request;\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({ 'nonce': nonce }, query));\n            let secret = this.base64ToBinary (this.secret);\n            let auth = request + '\\0' + body;\n            let signature = this.hmac (this.encode (auth), secret, 'sha512', 'base64');\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Rest-Key': this.apiKey,\n                'Rest-Sign': this.decode (signature),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('result' in response)\n            if (response['result'] === 'success')\n                return response;\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","\"use strict\";\n\n//-----------------------------------------------------------------------------\n\nconst isNode    = (typeof window === 'undefined') && !(typeof WorkerGlobalScope !== 'undefined' && self instanceof WorkerGlobalScope)\n    , functions = require ('./functions')\n    , throttle  = require ('./throttle')\n    , defaultFetch = isNode ? require ('fetch-ponyfill')().fetch : fetch\n    , Market    = require ('./Market')\n\nconst { deepExtend\n      , extend\n      , sleep\n      , timeout\n      , flatten\n      , indexBy\n      , sortBy\n      , groupBy\n      , aggregate\n      , uuid\n      , precisionFromString } = functions\n\nconst { ExchangeError\n      , NotSupported\n      , AuthenticationError\n      , DDoSProtection\n      , RequestTimeout\n      , ExchangeNotAvailable } = require ('./errors')\n\n// stub until we get a better solution for Webpack and React\n// const journal = isNode && require ('./journal')\nconst journal = undefined\n\nmodule.exports = class Exchange {\n\n    getMarket (symbol) {\n\n        if (!this.marketClasses)\n            this.marketClasses = {}\n\n        let marketClass = this.marketClasses[symbol]\n\n        if (marketClass)\n            return marketClass\n\n        marketClass = new Market (this, symbol)\n        this.marketClasses[symbol] = marketClass // only one Market instance per market\n        return marketClass\n    }\n\n    describe () { return {} }\n\n    constructor (userConfig = {}) {\n\n        Object.assign (this, functions, { encode: string => string, decode: string => string })\n\n        if (isNode)\n            this.nodeVersion = process.version.match (/\\d+\\.\\d+.\\d+/) [0]\n\n        // this.initRestRateLimiter ()\n\n        // if (isNode) {\n        //     this.userAgent = {\n        //         'User-Agent': 'ccxt/' + Exchange.ccxtVersion +\n        //             ' (+https://github.com/ccxt/ccxt)' +\n        //             ' Node.js/' + this.nodeVersion + ' (JavaScript)'\n        //     }\n        // }\n\n        this.userAgents = {\n            'chrome': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/62.0.3202.94 Safari/537.36',\n            'chrome39': 'Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/39.0.2171.71 Safari/537.36',\n        }\n\n        this.headers = {}\n\n        // prepended to URL, like https://proxy.com/https://exchange.com/api...\n        this.proxy = ''\n        this.origin = '*' // CORS origin\n\n        this.iso8601          = timestamp => new Date (timestamp).toISOString ()\n        this.parse8601        = x => Date.parse (((x.indexOf ('+') >= 0) || (x.slice (-1) == 'Z')) ? x : (x + 'Z'))\n        this.milliseconds     = Date.now\n        this.microseconds     = () => Math.floor (this.milliseconds () * 1000)\n        this.seconds          = () => Math.floor (this.milliseconds () / 1000)\n        this.id               = undefined\n\n        // rate limiter settings\n        this.enableRateLimit  = false\n        this.rateLimit        = 2000  // milliseconds = seconds * 1000\n\n        this.parseJsonResponse             = true  // whether a reply is required to be in JSON or not\n        this.substituteCommonCurrencyCodes = true  // reserved\n        this.parseBalanceFromOpenOrders    = false // some exchanges return balance updates from order API endpoints\n\n        // do not delete this line, it is needed for users to be able to define their own fetchImplementation\n        this.fetchImplementation = defaultFetch\n\n        this.timeout          = 10000 // milliseconds\n        this.verbose          = false\n        this.debug            = false\n        this.journal          = 'debug.json'\n        this.userAgent        = undefined\n        this.twofa            = false // two-factor authentication (2FA)\n        this.timeframes       = undefined\n        this.hasPublicAPI         = true\n        this.hasPrivateAPI        = true\n        this.hasCORS              = false\n        this.hasDeposit           = false\n        this.hasFetchBalance      = true\n        this.hasFetchClosedOrders = false\n        this.hasFetchCurrencies   = false\n        this.hasFetchMyTrades     = false\n        this.hasFetchOHLCV        = false\n        this.hasFetchOpenOrders   = false\n        this.hasFetchOrder        = false\n        this.hasFetchOrderBook    = true\n        this.hasFetchOrders       = false\n        this.hasFetchTicker       = true\n        this.hasFetchTickers      = false\n        this.hasFetchBidsAsks     = false\n        this.hasFetchTrades       = true\n        this.hasWithdraw          = false\n        this.hasCreateOrder       = this.hasPrivateAPI\n        this.hasCancelOrder       = this.hasPrivateAPI\n\n        this.apiKey   = undefined\n        this.secret   = undefined\n        this.uid      = undefined\n        this.login    = undefined\n        this.password = undefined\n\n        this.requiredCredentials = {\n            'apiKey':   true,\n            'secret':   true,\n            'uid':      false,\n            'login':    false,\n            'password': false,\n        }\n\n        this.exceptions = {}\n        this.balance    = {}\n        this.orderbooks = {}\n        this.tickers    = {}\n        this.fees       = {}\n        this.orders     = {}\n        this.trades     = {}\n        this.currencies = {}\n\n        this.last_http_response = undefined\n        this.last_json_response = undefined\n\n        this.arrayConcat = (a, b) => a.concat (b)\n\n        // TODO: generate\n\n        this.market_id                   = this.marketId\n        this.market_ids                  = this.marketIds\n        this.array_concat                = this.arrayConcat\n        this.implode_params              = this.implodeParams\n        this.extract_params              = this.extractParams\n        this.fetch_balance               = this.fetchBalance\n        this.fetch_free_balance          = this.fetchFreeBalance\n        this.fetch_used_balance          = this.fetchUsedBalance\n        this.fetch_total_balance         = this.fetchTotalBalance\n        this.fetch_l2_order_book         = this.fetchL2OrderBook\n        this.fetch_order_book            = this.fetchOrderBook\n        this.fetch_bids_asks             = this.fetchBidsAsks\n        this.fetch_tickers               = this.fetchTickers\n        this.fetch_ticker                = this.fetchTicker\n        this.fetch_trades                = this.fetchTrades\n        this.fetch_order                 = this.fetchOrder\n        this.fetch_orders                = this.fetchOrders\n        this.fetch_open_orders           = this.fetchOpenOrders\n        this.fetch_closed_orders         = this.fetchClosedOrders\n        this.fetch_order_status          = this.fetchOrderStatus\n        this.fetch_markets               = this.fetchMarkets\n        this.load_markets                = this.loadMarkets\n        this.set_markets                 = this.setMarkets\n        this.parse_balance               = this.parseBalance\n        this.parse_bid_ask               = this.parseBidAsk\n        this.parse_bids_asks             = this.parseBidsAsks\n        this.parse_order_book            = this.parseOrderBook\n        this.parse_trades                = this.parseTrades\n        this.parse_orders                = this.parseOrders\n        this.parse_ohlcv                 = this.parseOHLCV\n        this.parse_ohlcvs                = this.parseOHLCVs\n        this.edit_limit_buy_order        = this.editLimitBuyOrder\n        this.edit_limit_sell_order       = this.editLimitSellOrder\n        this.edit_limit_order            = this.editLimitOrder\n        this.edit_order                  = this.editOrder\n        this.create_limit_buy_order      = this.createLimitBuyOrder\n        this.create_limit_sell_order     = this.createLimitSellOrder\n        this.create_market_buy_order     = this.createMarketBuyOrder\n        this.create_market_sell_order    = this.createMarketSellOrder\n        this.create_order                = this.createOrder\n        this.calculate_fee               = this.calculateFee\n        this.common_currency_code        = this.commonCurrencyCode\n        this.price_to_precision          = this.priceToPrecision\n        this.amount_to_precision         = this.amountToPrecision\n        this.amount_to_string            = this.amountToString\n        this.fee_to_precision            = this.feeToPrecision\n        this.cost_to_precision           = this.costToPrecision\n        this.precisionFromString         = precisionFromString\n        this.precision_from_string       = precisionFromString\n        this.truncate                    = functions.truncate\n        this.truncate_to_string          = functions.truncate_to_string\n        this.uuid                        = uuid\n\n        // API methods metainfo\n        this.has = {\n            'cancelOrder': this.hasPrivateAPI,\n            'createDepositAddress': false,\n            'createOrder': this.hasPrivateAPI,\n            'deposit': false,\n            'fetchBalance': this.hasPrivateAPI,\n            'fetchClosedOrders': false,\n            'fetchCurrencies': false,\n            'fetchDepositAddress': false,\n            'fetchMarkets': true,\n            'fetchMyTrades': false,\n            'fetchOHLCV': false,\n            'fetchOpenOrders': false,\n            'fetchOrder': false,\n            'fetchOrderBook': true,\n            'fetchOrders': false,\n            'fetchTicker': true,\n            'fetchTickers': false,\n            'fetchBidsAsks': false,\n            'fetchTrades': true,\n            'withdraw': false,\n        }\n\n        // merge configs\n        const config = deepExtend (this.describe (), userConfig)\n\n        // merge to this\n        for (const [property, value] of Object.entries (config))\n            this[property] = deepExtend (this[property], value)\n\n        if (this.api)\n            this.defineRestApi (this.api, 'request')\n\n        this.initRestRateLimiter ()\n\n        if (this.markets)\n            this.setMarkets (this.markets)\n\n        if (this.debug && journal) {\n            journal (() => this.journal, this, Object.keys (this.has))\n        }\n    }\n\n    defaults () {\n        return { /* override me */ }\n    }\n\n    nonce () {\n        return this.seconds ()\n    }\n\n    encodeURIComponent (...args) {\n        return encodeURIComponent (...args)\n    }\n\n    checkRequiredCredentials () {\n        Object.keys (this.requiredCredentials).map (key => {\n            if (this.requiredCredentials[key] && !this[key])\n                throw new AuthenticationError (this.id + ' requires `' + key + '`')\n        })\n    }\n\n    initRestRateLimiter () {\n\n        const fetchImplementation = this.fetchImplementation\n\n        this.tokenBucket = this.extend ({\n            refillRate:  1 / this.rateLimit,\n            delay:       1,\n            capacity:    1,\n            defaultCost: 1,\n            maxCapacity: 1000,\n        }, this.tokenBucket)\n\n        this.throttle = throttle (this.tokenBucket)\n\n        this.executeRestRequest = function (url, method = 'GET', headers = undefined, body = undefined) {\n\n            let promise =\n                fetchImplementation (url, { 'method': method, 'headers': headers, 'body': body, 'agent': this.tunnelAgent || null, timeout: this.timeout})\n                    .catch (e => {\n                        if (isNode)\n                            throw new ExchangeNotAvailable ([ this.id, method, url, e.type, e.message ].join (' '))\n                        throw e // rethrow all unknown errors\n                    })\n                    .then (response => this.handleRestErrors (response, url, method, headers, body))\n                    .then (response => this.handleRestResponse (response, url, method, headers, body))\n\n            return timeout (this.timeout, promise).catch (e => {\n                if (e instanceof RequestTimeout)\n                    throw new RequestTimeout (this.id + ' ' + method + ' ' + url + ' ' + e.message + ' (' + this.timeout + ' ms)')\n                throw e\n            })\n        }\n    }\n\n    defineRestApi (api, methodName, options = {}) {\n\n        for (const type of Object.keys (api)) {\n            for (const httpMethod of Object.keys (api[type])) {\n\n                let urls = api[type][httpMethod]\n                for (let i = 0; i < urls.length; i++) {\n                    let url = urls[i].trim ()\n                    let splitPath = url.split (/[^a-zA-Z0-9]/)\n\n                    let uppercaseMethod  = httpMethod.toUpperCase ()\n                    let lowercaseMethod  = httpMethod.toLowerCase ()\n                    let camelcaseMethod  = this.capitalize (lowercaseMethod)\n                    let camelcaseSuffix  = splitPath.map (this.capitalize).join ('')\n                    let underscoreSuffix = splitPath.map (x => x.trim ().toLowerCase ()).filter (x => x.length > 0).join ('_')\n\n                    if (camelcaseSuffix.indexOf (camelcaseMethod) === 0)\n                        camelcaseSuffix = camelcaseSuffix.slice (camelcaseMethod.length)\n\n                    if (underscoreSuffix.indexOf (lowercaseMethod) === 0)\n                        underscoreSuffix = underscoreSuffix.slice (lowercaseMethod.length)\n\n                    let camelcase  = type + camelcaseMethod + this.capitalize (camelcaseSuffix)\n                    let underscore = type + '_' + lowercaseMethod + '_' + underscoreSuffix\n\n                    if ('suffixes' in options) {\n                        if ('camelcase' in options['suffixes'])\n                            camelcase += options['suffixes']['camelcase']\n                        if ('underscore' in options.suffixes)\n                            underscore += options['suffixes']['underscore']\n                    }\n\n                    if ('underscore_suffix' in options)\n                        underscore += options.underscoreSuffix;\n                    if ('camelcase_suffix' in options)\n                        camelcase += options.camelcaseSuffix;\n\n                    let partial = async params => this[methodName] (url, type, uppercaseMethod, params || {})\n\n                    this[camelcase]  = partial\n                    this[underscore] = partial\n                }\n            }\n        }\n    }\n\n    fetch (url, method = 'GET', headers = undefined, body = undefined) {\n\n        if (isNode && this.userAgent) {\n            if (typeof this.userAgent == 'string')\n                headers = extend ({ 'User-Agent': this.userAgent }, headers)\n            else if ((typeof this.userAgent == 'object') && ('User-Agent' in this.userAgent))\n                headers = extend (this.userAgent, headers)\n        }\n\n        if (typeof this.proxy == 'function') {\n\n            url = this.proxy (url)\n            if (isNode)\n                headers = extend ({ 'Origin': this.origin }, headers)\n\n        } else if (typeof this.proxy == 'string') {\n\n            if (this.proxy.length)\n                if (isNode)\n                    headers = extend ({ 'Origin': this.origin }, headers)\n\n            url = this.proxy + url\n        }\n\n        headers = extend (this.headers, headers)\n\n        if (this.verbose)\n            console.log (this.id, method, url, \"\\nRequest:\\n\", headers, body)\n\n        return this.executeRestRequest (url, method, headers, body)\n    }\n\n    async fetch2 (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n\n        if (this.enableRateLimit)\n            await this.throttle ()\n\n        let request = this.sign (path, api, method, params, headers, body)\n        return this.fetch (request.url, request.method, request.headers, request.body)\n    }\n\n    request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        return this.fetch2 (path, api, method, params, headers, body)\n    }\n\n    handleErrors (statusCode, statusText, url, method, headers, body) {\n        // override me\n    }\n\n    defaultErrorHandler (code, reason, url, method, headers, body) {\n        if ((code >= 200) && (code <= 300))\n            return body\n        let error = undefined\n        this.last_http_response = body\n        let details = body\n        let match = body.match (/\\<title\\>([^<]+)/i)\n        if (match)\n            details = match[1].trim ();\n        if ([ 418, 429 ].includes (code)) {\n            error = DDoSProtection\n        } else if ([ 404, 409, 500, 501, 502, 520, 521, 522, 525 ].includes (code)) {\n            error = ExchangeNotAvailable\n        } else if ([ 400, 403, 405, 503, 530 ].includes (code)) {\n            let ddosProtection = body.match (/cloudflare|incapsula/i)\n            if (ddosProtection) {\n                error = DDoSProtection\n            } else {\n                error = ExchangeNotAvailable\n                details += ' (possible reasons: ' + [\n                    'invalid API keys',\n                    'bad or old nonce',\n                    'exchange is down or offline',\n                    'on maintenance',\n                    'DDoS protection',\n                    'rate-limiting',\n                ].join (', ') + ')'\n            }\n        } else if ([ 408, 504 ].includes (code)) {\n            error = RequestTimeout\n        } else if ([ 401, 511 ].includes (code)) {\n            error = AuthenticationError\n        } else {\n            error = ExchangeError\n        }\n        throw new error ([ this.id, method, url, code, reason, details ].join (' '))\n    }\n\n    handleRestErrors (response, url, method = 'GET', headers = undefined, body = undefined) {\n\n        if (typeof response === 'string')\n            return response\n\n        return response.text ().then (text => {\n\n            const args = [ response.status, response.statusText, url, method, headers, text ]\n\n            if (this.verbose)\n                console.log (this.id, method, url, response.status, response.statusText, headers, text ? (\"\\nResponse:\\n\" + text) : '')\n\n            this.handleErrors (...args)\n            return this.defaultErrorHandler (...args)\n        })\n    }\n\n    handleRestResponse (response, url, method = 'GET', headers = undefined, body = undefined) {\n\n        try {\n\n            this.last_http_response = response\n            if (this.parseJsonResponse) {\n                this.last_json_response =\n                    ((typeof response === 'string') && (response.length > 1)) ?\n                        JSON.parse (response) : response\n                return this.last_json_response\n            }\n\n            return response\n\n        } catch (e) {\n\n            let maintenance = response.match (/offline|busy|retry|wait|unavailable|maintain|maintenance|maintenancing/i)\n            let ddosProtection = response.match (/cloudflare|incapsula|overload/i)\n\n            if (e instanceof SyntaxError) {\n\n                let error = ExchangeNotAvailable\n                let details = 'not accessible from this location at the moment'\n                if (maintenance)\n                    details = 'offline, on maintenance or unreachable from this location at the moment'\n                if (ddosProtection)\n                    error = DDoSProtection\n                throw new error ([ this.id, method, url, details ].join (' '))\n            }\n\n            if (this.verbose)\n                console.log (this.id, method, url, 'error', e, \"response body:\\n'\" + response + \"'\")\n\n            throw e\n        }\n    }\n\n    setMarkets (markets, currencies = undefined) {\n        let values = Object.values (markets).map (market => deepExtend ({\n            'limits': this.limits,\n            'precision': this.precision,\n        }, this.fees['trading'], market))\n        this.markets = deepExtend (this.markets, indexBy (values, 'symbol'))\n        this.marketsById = indexBy (markets, 'id')\n        this.markets_by_id = this.marketsById\n        this.symbols = Object.keys (this.markets).sort ()\n        this.ids = Object.keys (this.markets_by_id).sort ()\n        if (currencies) {\n            this.currencies = deepExtend (currencies, this.currencies)\n        } else {\n            const baseCurrencies =\n                values.filter (market => 'base' in market)\n                    .map (market => ({\n                        id: market.baseId || market.base,\n                        code: market.base,\n                        precision: market.precision ? (market.precision.base || market.precision.amount) : 8,\n                    }))\n            const quoteCurrencies =\n                values.filter (market => 'quote' in market)\n                    .map (market => ({\n                        id: market.quoteId || market.quote,\n                        code: market.quote,\n                        precision: market.precision ? (market.precision.quote || market.precision.price) : 8,\n                    }))\n            const allCurrencies = baseCurrencies.concat (quoteCurrencies)\n            const groupedCurrencies = groupBy (allCurrencies, 'code')\n            const currencies = Object.keys (groupedCurrencies).map (code =>\n                groupedCurrencies[code].reduce ((previous, current) =>\n                    ((previous.precision > current.precision) ? previous : current), groupedCurrencies[code][0]))\n            const sortedCurrencies = sortBy (flatten (currencies), 'code')\n            this.currencies = deepExtend (indexBy (sortedCurrencies, 'code'), this.currencies)\n        }\n        this.currencies_by_id = indexBy (this.currencies, 'id')\n        return this.markets\n    }\n\n    async loadMarkets (reload = false) {\n        if (!reload && this.markets) {\n            if (!this.marketsById) {\n                return this.setMarkets (this.markets)\n            }\n            return this.markets\n        }\n        const markets = await this.fetchMarkets ()\n        let currencies = undefined\n        if (this.has.fetchCurrencies) {\n            currencies = await this.fetchCurrencies ()\n        }\n        return this.setMarkets (markets, currencies)\n    }\n\n    fetchBidsAsks (symbols = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchBidsAsks not supported yet')\n    }\n\n    fetchTickers (symbols = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchTickers not supported yet')\n    }\n\n    fetchOrder (id, symbol = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchOrder not supported yet');\n    }\n\n    fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchOrders not supported yet');\n    }\n\n    fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchOpenOrders not supported yet');\n    }\n\n    fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchClosedOrders not supported yet');\n    }\n\n    fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchMyTrades not supported yet');\n    }\n\n    fetchCurrencies () {\n        throw new NotSupported (this.id + ' fetchCurrencies not supported yet');\n    }\n\n    fetchMarkets () {\n        return new Promise ((resolve, reject) => resolve (this.markets))\n    }\n\n    async fetchOrderStatus (id, market = undefined) {\n        let order = await this.fetchOrder (id)\n        return order['status']\n    }\n\n    account () {\n        return {\n            'free': 0.0,\n            'used': 0.0,\n            'total': 0.0,\n        }\n    }\n\n    commonCurrencyCode (currency) {\n        if (!this.substituteCommonCurrencyCodes)\n            return currency\n        if (currency == 'XBT')\n            return 'BTC'\n        if (currency == 'BCC')\n            return 'BCH'\n        if (currency == 'DRK')\n            return 'DASH'\n        return currency\n    }\n\n    currency (code) {\n\n        if (typeof this.currencies == 'undefined')\n            return new ExchangeError (this.id + ' currencies not loaded')\n\n        if ((typeof code === 'string') && (code in this.currencies))\n            return this.currencies[code]\n\n        throw new ExchangeError (this.id + ' does not have currency code ' + code)\n    }\n\n\n    market (symbol) {\n\n        if (typeof this.markets == 'undefined')\n            return new ExchangeError (this.id + ' markets not loaded')\n\n        if ((typeof symbol === 'string') && (symbol in this.markets))\n            return this.markets[symbol]\n\n        throw new ExchangeError (this.id + ' does not have market symbol ' + symbol)\n    }\n\n    marketId (symbol) {\n        return this.market (symbol).id || symbol\n    }\n\n    marketIds (symbols) {\n        return symbols.map (symbol => this.marketId(symbol));\n    }\n\n    symbol (symbol) {\n        return this.market (symbol).symbol || symbol\n    }\n\n    extractParams (string) {\n        let re = /{([a-zA-Z0-9_]+?)}/g\n        let matches = []\n        let match\n        while (match = re.exec (string))\n            matches.push (match[1])\n        return matches\n    }\n\n    implodeParams (string, params) {\n        for (let property in params)\n            string = string.replace ('{' + property + '}', params[property])\n        return string\n    }\n\n    url (path, params = {}) {\n        let result = this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path))\n        if (Object.keys (query).length)\n            result += '?' + this.urlencode (query)\n        return result\n    }\n\n    parseBidAsk (bidask, priceKey = 0, amountKey = 1) {\n        let price = parseFloat (bidask[priceKey])\n        let amount = parseFloat (bidask[amountKey])\n        return [ price, amount ]\n    }\n\n    parseBidsAsks (bidasks, priceKey = 0, amountKey = 1) {\n        return Object.values (bidasks || []).map (bidask => this.parseBidAsk (bidask, priceKey, amountKey))\n    }\n\n    async fetchL2OrderBook (symbol, params = {}) {\n        let orderbook = await this.fetchOrderBook (symbol, params)\n        return extend (orderbook, {\n            'bids': aggregate (orderbook.bids),\n            'asks': aggregate (orderbook.asks),\n        })\n    }\n\n    parseOrderBook (orderbook, timestamp = undefined, bidsKey = 'bids', asksKey = 'asks', priceKey = 0, amountKey = 1) {\n        timestamp = timestamp || this.milliseconds ();\n        return {\n            'bids': sortBy ((bidsKey in orderbook) ? this.parseBidsAsks (orderbook[bidsKey], priceKey, amountKey) : [], 0, true),\n            'asks': sortBy ((asksKey in orderbook) ? this.parseBidsAsks (orderbook[asksKey], priceKey, amountKey) : [], 0),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n        }\n    }\n\n    getCurrencyUsedOnOpenOrders (currency) {\n        return Object.values (this.orders).filter (order => (order['status'] == 'open')).reduce ((total, order) => {\n            let symbol = order['symbol'];\n            let market = this.markets[symbol];\n            let amount = order['remaining']\n            if (currency == market['base'] && order['side'] == 'sell') {\n                return total + amount\n            } else if (currency == market['quote'] && order['side'] == 'buy') {\n                return total + (order['cost'] || (order['price'] * amount))\n            } else {\n                return total\n            }\n        }, 0)\n    }\n\n    parseBalance (balance) {\n\n        const currencies = Object.keys (this.omit (balance, 'info'));\n\n        currencies.forEach (currency => {\n\n            if (typeof balance[currency].used == 'undefined') {\n\n                if (this.parseBalanceFromOpenOrders && ('open_orders' in balance['info'])) {\n                    const exchangeOrdersCount = balance['info']['open_orders'];\n                    const cachedOrdersCount = Object.values (this.orders).filter (order => (order['status'] == 'open')).length;\n                    if (cachedOrdersCount == exchangeOrdersCount) {\n                        balance[currency].used = this.getCurrencyUsedOnOpenOrders (currency)\n                        balance[currency].total = balance[currency].used + balance[currency].free\n                    }\n                } else {\n                    balance[currency].used = this.getCurrencyUsedOnOpenOrders (currency)\n                    balance[currency].total = balance[currency].used + balance[currency].free\n                }\n            }\n\n            [ 'free', 'used', 'total' ].forEach (account => {\n                balance[account] = balance[account] || {}\n                balance[account][currency] = balance[currency][account]\n            })\n        })\n\n        return balance\n    }\n\n    async fetchPartialBalance (part, params = {}) {\n        let balance = await this.fetchBalance (params)\n        return balance[part]\n    }\n\n    fetchFreeBalance (params = {}) {\n        return this.fetchPartialBalance ('free', params)\n    }\n\n    fetchUsedBalance (params = {}) {\n        return this.fetchPartialBalance ('used', params)\n    }\n\n    fetchTotalBalance (params = {}) {\n        return this.fetchPartialBalance ('total', params)\n    }\n\n    filterBySinceLimit (array, since = undefined, limit = undefined) {\n        if (since)\n            array = array.filter (entry => entry.timestamp > since)\n        if (limit)\n            array = array.slice (0, limit)\n        return array\n    }\n\n    parseTrades (trades, market = undefined, since = undefined, limit = undefined) {\n        let result = Object.values (trades).map (trade => this.parseTrade (trade, market))\n        result = sortBy (result, 'timestamp', true)\n        return this.filterBySinceLimit (result, since, limit)\n    }\n\n    parseOrders (orders, market = undefined, since = undefined, limit = undefined) {\n        let result = Object.values (orders).map (order => this.parseOrder (order, market))\n        return this.filterBySinceLimit (result, since, limit)\n    }\n\n    filterOrdersBySymbol (orders, symbol = undefined) {\n        let grouped = this.groupBy (orders, 'symbol')\n        if (symbol) {\n            if (symbol in grouped)\n                return grouped[symbol]\n            return []\n        }\n        return orders\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return ohlcv\n    }\n\n    parseOHLCVs (ohlcvs, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        ohlcvs = Object.values (ohlcvs)\n        let result = []\n        for (let i = 0; i < ohlcvs.length; i++) {\n            if (limit && (result.length >= limit))\n                break;\n            let ohlcv = this.parseOHLCV (ohlcvs[i], market, timeframe, since, limit)\n            if (since && (ohlcv[0] < since))\n                continue\n            result.push (ohlcv)\n        }\n        return result\n    }\n\n    editLimitBuyOrder (id, symbol, ...args) {\n        return this.editLimitOrder (id, symbol, 'buy', ...args)\n    }\n\n    editLimitSellOrder (id, symbol, ...args) {\n        return this.editLimitOrder (id, symbol, 'sell', ...args)\n    }\n\n    editLimitOrder (id, symbol, ...args) {\n        return this.editOrder (id, symbol, 'limit', ...args)\n    }\n\n    async editOrder (id, symbol, ...args) {\n        if (!this.enableRateLimit)\n            throw new ExchangeError (this.id + ' editOrder() requires enableRateLimit = true')\n        await this.cancelOrder (id, symbol);\n        return this.createOrder (symbol, ...args)\n    }\n\n    createLimitBuyOrder (symbol, ...args) {\n        return this.createOrder  (symbol, 'limit', 'buy', ...args)\n    }\n\n    createLimitSellOrder (symbol, ...args) {\n        return this.createOrder (symbol, 'limit', 'sell', ...args)\n    }\n\n    createMarketBuyOrder (symbol, amount, params = {}) {\n        return this.createOrder (symbol, 'market', 'buy', amount, undefined, params)\n    }\n\n    createMarketSellOrder (symbol, amount, params = {}) {\n        return this.createOrder (symbol, 'market', 'sell', amount, undefined, params)\n    }\n\n    costToPrecision (symbol, cost) {\n        return parseFloat (cost).toFixed (this.markets[symbol].precision.price)\n    }\n\n    priceToPrecision (symbol, price) {\n        return parseFloat (price).toFixed (this.markets[symbol].precision.price)\n    }\n\n    amountToPrecision (symbol, amount) {\n        return this.truncate (amount, this.markets[symbol].precision.amount)\n    }\n\n    amountToString (symbol, amount) {\n        return this.truncate_to_string (amount, this.markets[symbol].precision.amount)\n    }\n\n    amountToLots (symbol, amount) {\n        return this.amountToPrecision (symbol, Math.floor (amount / this.markets[symbol].lot) * this.markets[symbol].lot)\n    }\n\n    feeToPrecision (symbol, fee) {\n        return parseFloat (fee).toFixed (this.markets[symbol].precision.price)\n    }\n\n    calculateFee (symbol, type, side, amount, price, takerOrMaker = 'taker', params = {}) {\n        let market = this.markets[symbol]\n        let rate = market[takerOrMaker]\n        let cost = parseFloat (this.costToPrecision (symbol, amount * price))\n        return {\n            'type': takerOrMaker,\n            'currency': market['quote'],\n            'rate': rate,\n            'cost': parseFloat (this.feeToPrecision (symbol, rate * cost)),\n        }\n    }\n\n    Ymd (timestamp, infix = ' ') {\n        let date = new Date (timestamp)\n        let Y = date.getUTCFullYear ()\n        let m = date.getUTCMonth () + 1\n        let d = date.getUTCDate ()\n        m = m < 10 ? ('0' + m) : m\n        d = d < 10 ? ('0' + d) : d\n        return Y + '-' + m + '-' + d\n    }\n\n    YmdHMS (timestamp, infix = ' ') {\n        let date = new Date (timestamp)\n        let Y = date.getUTCFullYear ()\n        let m = date.getUTCMonth () + 1\n        let d = date.getUTCDate ()\n        let H = date.getUTCHours ()\n        let M = date.getUTCMinutes ()\n        let S = date.getUTCSeconds ()\n        m = m < 10 ? ('0' + m) : m\n        d = d < 10 ? ('0' + d) : d\n        H = H < 10 ? ('0' + H) : H\n        M = M < 10 ? ('0' + M) : M\n        S = S < 10 ? ('0' + S) : S\n        return Y + '-' + m + '-' + d + infix + H + ':' + M + ':' + S\n    }\n}\n","\"use strict\";\n\nmodule.exports = class Market {\n\n    constructor (exchange, symbol) {\n        this.exchange = exchange;\n        this.symbol = symbol;\n        this.market = exchange.markets[symbol];\n    }\n\n    amountToPrecision (amount) {\n        return this.exchange.amountToPrecision (this.symbol, amount)\n    }\n\n    createLimitBuyOrder(amount, price) {\n        return this.exchange.createLimitBuyOrder (this.symbol, amount, price)\n    }\n\n    createLimitSellOrder(amount, price) {\n        return this.exchange.createLimitSellOrder (this.symbol, amount, price)\n    }\n}\n","class BaseError extends Error {\n    constructor (message) {\n        super (message)\n        // a workaround to make `instanceof BaseError` work in ES5\n        this.constructor = BaseError\n        this.__proto__   = BaseError.prototype\n        this.message     = message\n    }\n}\n\nclass ExchangeError extends BaseError {\n    constructor (message) {\n        super (message)\n        this.constructor = ExchangeError\n        this.__proto__   = ExchangeError.prototype\n        this.message     = message\n    }\n}\n\nclass NotSupported extends ExchangeError {\n    constructor (message) {\n        super (message)\n        this.constructor = NotSupported\n        this.__proto__   = NotSupported.prototype\n        this.message     = message\n    }\n}\n\nclass AuthenticationError extends ExchangeError {\n    constructor (message) {\n        super (message)\n        this.constructor = AuthenticationError\n        this.__proto__   = AuthenticationError.prototype\n        this.message     = message\n    }\n}\n\nclass InvalidNonce extends ExchangeError {\n    constructor (message) {\n        super (message)\n        this.constructor = InvalidNonce\n        this.__proto__   = InvalidNonce.prototype\n        this.message     = message\n    }\n}\n\nclass InsufficientFunds extends ExchangeError {\n    constructor (message) {\n        super (message)\n        this.constructor = InsufficientFunds\n        this.__proto__   = InsufficientFunds.prototype\n        this.message     = message\n    }\n}\n\nclass InvalidOrder extends ExchangeError {\n    constructor (message) {\n        super (message)\n        this.constructor = InvalidOrder\n        this.__proto__   = InvalidOrder.prototype\n        this.message     = message\n    }\n}\n\nclass OrderNotFound extends InvalidOrder {\n    constructor (message) {\n        super (message)\n        this.constructor = OrderNotFound\n        this.__proto__   = OrderNotFound.prototype\n        this.message     = message\n    }\n}\n\nclass OrderNotCached extends InvalidOrder {\n    constructor (message) {\n        super (message)\n        this.constructor = OrderNotCached\n        this.__proto__   = OrderNotCached.prototype\n        this.message     = message\n    }\n}\n\nclass CancelPending extends InvalidOrder {\n    constructor (message) {\n        super (message)\n        this.constructor = CancelPending\n        this.__proto__   = CancelPending.prototype\n        this.message     = message\n    }\n}\n\nclass NetworkError extends BaseError {\n    constructor (message) {\n        super (message)\n        this.constructor = NetworkError\n        this.__proto__   = NetworkError.prototype\n        this.message     = message\n    }\n}\n\nclass DDoSProtection extends NetworkError {\n    constructor (message) {\n        super (message)\n        this.constructor = DDoSProtection\n        this.__proto__   = DDoSProtection.prototype\n        this.message     = message\n    }\n}\n\nclass RequestTimeout extends NetworkError {\n    constructor (message) {\n        super (message)\n        this.constructor = RequestTimeout\n        this.__proto__   = RequestTimeout.prototype\n        this.message     = message\n    }\n}\n\nclass ExchangeNotAvailable extends NetworkError {\n    constructor (message) {\n        super (message)\n        this.constructor = ExchangeNotAvailable\n        this.__proto__   = ExchangeNotAvailable.prototype\n        this.message     = message\n    }\n}\n\nmodule.exports = {\n\n    BaseError,\n    ExchangeError,\n    NotSupported,\n    AuthenticationError,\n    InvalidNonce,\n    InsufficientFunds,\n    InvalidOrder,\n    OrderNotFound,\n    OrderNotCached,\n    CancelPending,\n    NetworkError,\n    DDoSProtection,\n    RequestTimeout,\n    ExchangeNotAvailable,\n}\n","\"use strict\";\n\n//-----------------------------------------------------------------------------\n\nconst CryptoJS = require ('crypto-js')\n    , qs       = require ('qs') // querystring\n\n//-----------------------------------------------------------------------------\n\nconst { RequestTimeout } = require ('./errors')\n\n//-----------------------------------------------------------------------------\n// utility helpers\n\nconst setTimeout_original = setTimeout\n\n// setTimeout can fire earlier than specified, so we need to ensure it does not happen...\n\nconst setTimeout_safe = (done, ms, setTimeout = setTimeout_original /* overrideable for mocking purposes */, targetTime = Date.now () + ms) => {\n\n    let clearInnerTimeout = () => {}\n    let active = true\n\n    let id = setTimeout (() => {\n        active = true\n        const rest = targetTime - Date.now ()\n        if (rest > 0) {\n            clearInnerTimeout = setTimeout_safe (done, rest, setTimeout, targetTime) // try sleep more\n        } else {\n            done ()\n        }\n    }, ms)\n\n    return function clear () {\n        if (active) {\n            active = false // dunno if IDs are unique on various platforms, so it's better to rely on this flag to exclude the possible cancellation of the wrong timer (if called after completion)\n            clearTimeout (id)\n        }\n        clearInnerTimeout ()\n    }\n}\n\nconst sleep = ms => new Promise (resolve => setTimeout_safe (resolve, ms))\n\nconst decimal = float => parseFloat (float).toString ()\n\nconst timeout = async (ms, promise) => {\n\n    let clear = () => {}\n    const timeout = new Promise (resolve => (clear = setTimeout_safe (resolve, ms)))\n\n    try {\n        return await Promise.race ([promise, timeout.then (() => { throw new RequestTimeout ('request timed out') })])\n    } finally {\n        clear () // fixes https://github.com/ccxt/ccxt/issues/749\n    }\n}\n\nconst capitalize = string => string.length ? (string.charAt (0).toUpperCase () + string.slice (1)) : string\n\nconst keysort = object => {\n    const result = {}\n    Object.keys (object).sort ().forEach (key => result[key] = object[key])\n    return result\n}\n\nconst extend = (...args) => Object.assign ({}, ...args)\n\nconst deepExtend = function (...args) {\n\n    // if (args.length < 1)\n    //     return args\n    // else if (args.length < 2)\n    //     return args[0]\n\n    let result = undefined\n\n    for (const arg of args) {\n\n        if (arg && (typeof arg === 'object') && (arg.constructor === Object || !('constructor' in arg))) {\n\n            if (typeof result !== 'object') {\n                result = {}\n            }\n\n            for (const key in arg) {\n                result[key] = deepExtend (result[key], arg[key])\n            }\n\n        } else {\n\n            result = arg\n        }\n    }\n\n    return result\n}\n\nconst omit = (object, ...args) => {\n    const result = extend (object)\n    for (const x of args) {\n        if (typeof x === 'string') {\n            delete result[x]\n        } else if (Array.isArray (x)) {\n            for (const k of x)\n                delete result[k]\n        }\n    }\n    return result\n}\n\nconst groupBy = (array, key) => {\n    const result = {}\n    Object\n        .values (array)\n        .filter (entry => typeof entry[key] != 'undefined')\n        .forEach (entry => {\n            if (typeof result[entry[key]] == 'undefined')\n                result[entry[key]] = []\n            result[entry[key]].push (entry)\n        })\n    return result\n}\n\nconst filterBy = (array, key, value = undefined) => {\n    if (value) {\n        let grouped = groupBy (array, key)\n        if (value in grouped)\n            return grouped[value]\n        return []\n    }\n    return array\n}\n\nconst indexBy = (array, key) => {\n    const result = {}\n    Object\n        .values (array)\n        .filter (entry => typeof entry[key] != 'undefined')\n        .forEach (entry => {\n            result[entry[key]] = entry\n        })\n    return result\n}\n\nconst sortBy = (array, key, descending = false) => {\n    descending = descending ? -1 : 1\n    return array.sort ((a, b) => ((a[key] < b[key]) ? -descending : ((a[key] > b[key]) ? descending : 0)))\n}\n\nconst flatten = (array, result = []) => {\n    for (let i = 0, length = array.length; i < length; i++) {\n        const value = array[i]\n        if (Array.isArray (value)) {\n            flatten (value, result)\n        } else {\n            result.push (value)\n        }\n    }\n    return result\n}\n\nconst unique = array => array.filter ((value, index, self) => (self.indexOf (value) == index))\n\nconst pluck = (array, key) => array\n                                .filter (element => (typeof element[key] != 'undefined'))\n                                .map (element => element[key])\n\nconst urlencode = object => qs.stringify (object)\nconst rawencode = object => qs.stringify (object, { encode: false })\n\nconst sum = (...args) => {\n    const result = args.filter (arg => typeof arg !== 'undefined')\n    return (result.length > 0) ?\n        result.reduce ((sum, value) => sum + value, 0) : undefined\n}\n\nconst safeFloat = (object, key, defaultValue = undefined) => {\n    if (key in object) {\n        if (typeof object[key] == 'number')\n            return object[key]\n        else if ((typeof object[key] == 'string') && object[key])\n            return parseFloat (object[key])\n    }\n    return defaultValue\n}\n\nconst safeString = (object, key, defaultValue = undefined) => {\n    if (!object || !(key in object))\n        return defaultValue;\n    let stringVal = object[key];\n    if (!stringVal && typeof stringVal != 'string' && typeof stringVal != 'number')\n        return defaultValue;\n    return stringVal.toString ();\n}\n\nconst safeInteger = (object, key, defaultValue = undefined) => {\n    if (!object || !(key in object))\n        return defaultValue;\n    let intVal = parseInt (object[key], 10);\n    return isNaN (intVal) ? defaultValue : intVal;\n}\n\nconst safeValue = (object, key, defaultValue = undefined) => {\n    return (object && (key in object) && object[key]) ? object[key] : defaultValue\n}\n\nconst uuid = a => a ?\n    (a ^ Math.random () * 16 >> a / 4).toString (16) :\n    ([1e7]+-1e3+-4e3+-8e3+-1e11).replace (/[018]/g, uuid)\n\n// See https://stackoverflow.com/questions/1685680/how-to-avoid-scientific-notation-for-large-numbers-in-javascript for discussion\n\nfunction toFixed (x) { // avoid scientific notation for too large and too small numbers\n\n    if (Math.abs (x) < 1.0) {\n        const e = parseInt (x.toString ().split ('e-')[1])\n        if (e) {\n            x *= Math.pow (10, e-1)\n            x = '0.' + (new Array (e)).join ('0') + x.toString ().substring (2)\n        }\n    } else {\n        let e = parseInt (x.toString ().split ('+')[1])\n        if (e > 20) {\n            e -= 20\n            x /= Math.pow (10, e)\n            x += (new Array (e+1)).join ('0')\n        }\n    }\n    return x\n}\n\n// See https://stackoverflow.com/questions/4912788/truncate-not-round-off-decimal-numbers-in-javascript for discussion\n\n// > So, after all it turned out, rounding bugs will always haunt you, no matter how hard you try to compensate them.\n// > Hence the problem should be attacked by representing numbers exactly in decimal notation.\n\nconst truncate_regExpCache = []\n    , truncate_to_string = (num, precision = 0) => {\n        num = toFixed (num)\n        if (precision > 0) {\n            const re = truncate_regExpCache[precision] || (truncate_regExpCache[precision] = new RegExp(\"([-]*\\\\d+\\\\.\\\\d{\" + precision + \"})(\\\\d)\"))\n            const [,result] = num.toString ().match (re) || [null, num]\n            return result.toString ()\n        }\n        return parseInt (num).toString ()\n    }\n    , truncate = (num, precision = 0) => parseFloat (truncate_to_string (num, precision))\n\nconst precisionFromString = (string) => {\n    const split = string.replace (/0+$/g, '').split ('.')\n    return (split.length > 1) ? (split[1].length) : 0\n}\n\nconst ordered = x => x // a stub to keep assoc keys in order, in JS it does nothing, it's mostly for Python\n\nconst aggregate = function (bidasks) {\n\n    let result = {}\n\n    bidasks.forEach (([ price, volume ]) => {\n        if (volume > 0)\n            result[price] = (result[price] || 0) + volume\n    })\n\n    return Object.keys (result).map (price => [\n        parseFloat (price),\n        parseFloat (result[price]),\n    ])\n}\n\n//-----------------------------------------------------------------------------\n// string ←→ binary ←→ base64 conversion routines\n\nconst stringToBinary = str => {\n    const arr = new Uint8Array (str.length)\n    for (let i = 0; i < str.length; i++) { arr[i] = str.charCodeAt(i); }\n    return CryptoJS.lib.WordArray.create (arr)\n}\n\nconst stringToBase64 = string => CryptoJS.enc.Latin1.parse (string).toString (CryptoJS.enc.Base64)\n    , utf16ToBase64  = string => CryptoJS.enc.Utf16 .parse (string).toString (CryptoJS.enc.Base64)\n    , base64ToBinary = string => CryptoJS.enc.Base64.parse (string)\n    , base64ToString = string => CryptoJS.enc.Base64.parse (string).toString (CryptoJS.enc.Utf8)\n    , binaryToString = string => string\n\nconst binaryConcat = (...args) => args.reduce ((a, b) => a.concat (b))\n\n// url-safe-base64 without equals signs, with + replaced by - and slashes replaced by underscores\nconst urlencodeBase64 = base64string => base64string.replace (/[=]+$/, '')\n                                                    .replace (/\\+/g, '-')\n                                                    .replace (/\\//g, '_')\n\n//-----------------------------------------------------------------------------\n// cryptography\n\nconst hash = (request, hash = 'md5', digest = 'hex') => {\n    const result = CryptoJS[hash.toUpperCase ()] (request)\n    return (digest == 'binary') ? result : result.toString (CryptoJS.enc[capitalize (digest)])\n}\n\nconst hmac = (request, secret, hash = 'sha256', digest = 'hex') => {\n    const encoding = (digest == 'binary') ? 'Latin1' : capitalize (digest)\n    return CryptoJS['Hmac' + hash.toUpperCase ()] (request, secret).toString (CryptoJS.enc[capitalize (encoding)])\n}\n\n//-----------------------------------------------------------------------------\n// a JSON Web Token authentication method\n\nconst jwt = (request, secret, alg = 'HS256', hash = 'sha256') => {\n    const encodedHeader = urlencodeBase64 (stringToBase64 (JSON.stringify ({ 'alg': alg, 'typ': 'JWT' })))\n        , encodedData = urlencodeBase64 (stringToBase64 (JSON.stringify (request)))\n        , token = [ encodedHeader, encodedData ].join ('.')\n        , signature = urlencodeBase64 (utf16ToBase64 (hmac (token, secret, hash, 'utf16')))\n    return [ token, signature ].join ('.')\n}\n\n//-----------------------------------------------------------------------------\n\nmodule.exports = {\n\n    setTimeout_safe,\n\n    // common utility functions\n\n    sleep,\n    timeout,\n    capitalize,\n    keysort,\n    extend,\n    deepExtend,\n    omit,\n    groupBy,\n    indexBy,\n    sortBy,\n    filterBy,\n    flatten,\n    unique,\n    pluck,\n    urlencode,\n    rawencode,\n    sum,\n    decimal,\n    safeFloat,\n    safeString,\n    safeInteger,\n    safeValue,\n    ordered,\n    aggregate,\n    truncate,\n    truncate_to_string,\n    uuid,\n    precisionFromString,\n\n    // underscore aliases\n\n    index_by: indexBy,\n    sort_by: sortBy,\n    group_by: groupBy,\n    filter_by: filterBy,\n    safe_float: safeFloat,\n    safe_string: safeString,\n    safe_integer: safeInteger,\n    safe_value: safeValue,\n\n    // crypto functions\n\n    binaryConcat,\n    stringToBinary,\n    binaryToString,\n    stringToBase64,\n    utf16ToBase64,\n    base64ToBinary,\n    base64ToString,\n    urlencodeBase64,\n    hash,\n    hmac,\n    jwt,\n\n    // json\n    json:   JSON.stringify,\n    unjson: JSON.parse\n}\n","\"use strict\";\n\nconst { sleep }  = require ('./functions')\n\nconst throttle = cfg => {\n\n    let lastTimestamp = Date.now ()\n        , numTokens = (typeof cfg.numTokens != 'undefined') ? cfg.numTokens : cfg.capacity\n        , queue = []\n        , running = false\n        , counter = 0\n\n    return Object.assign (cost => {\n\n        if (queue.length > cfg.maxCapacity)\n            throw new Error ('Backlog is over max capacity of ' + cfg.maxCapacity)\n\n        return new Promise (async (resolve, reject) => {\n\n            try {\n\n                queue.push ({ cost, resolve, reject })\n\n                if (!running) {\n                    running = true\n                    while (queue.length > 0) {\n                        const hasEnoughTokens = cfg.capacity ? (numTokens > 0) : (numTokens >= 0)\n                        if (hasEnoughTokens) {\n                            if (queue.length > 0) {\n                                let { cost, resolve, reject } = queue[0]\n                                cost = (cost || cfg.defaultCost)\n                                if (numTokens >= Math.min (cost, cfg.capacity)) {\n                                    numTokens -= cost\n                                    queue.shift ()\n                                    resolve ()\n                                }\n                            }\n                        }\n                        let now = Date.now ()\n                        let elapsed = now - lastTimestamp\n                        lastTimestamp = now\n                        numTokens = Math.min (cfg.capacity, numTokens + elapsed * cfg.refillRate)\n                        await sleep (cfg.delay)\n                    }\n                    running = false\n                }\n\n            } catch (e) {\n\n                reject (e)\n            }\n        })\n\n    }, cfg, {\n        configure: newCfg => throttle (Object.assign ({}, cfg, newCfg))\n    })\n}\n\nmodule.exports = throttle\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, AuthenticationError, DDoSProtection } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bibox extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bibox',\n            'name': 'Bibox',\n            'countries': [ 'CN', 'US', 'KR' ],\n            'version': 'v1',\n            'hasCORS': false,\n            'hasPublicAPI': false,\n            'hasFetchBalance': true,\n            'hasFetchCurrencies': true,\n            'hasFetchTickers': true,\n            'hasFetchOrders': true,\n            'hasFetchMyTrades': true,\n            'hasFetchOHLCV': true,\n            'hasWithdraw': true,\n            'has': {\n                'fetchBalance': true,\n                'fetchCurrencies': true,\n                'fetchTickers': true,\n                'fetchOrders': true,\n                'fetchMyTrades': true,\n                'fetchOHLCV': true,\n                'withdraw': true,\n            },\n            'timeframes': {\n                '1m': '1min',\n                '5m': '5min',\n                '15m': '15min',\n                '30m': '30min',\n                '1h': '1hour',\n                '8h': '12hour',\n                '1d': 'day',\n                '1w': 'week',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/34902611-2be8bf1a-f830-11e7-91a2-11b2f292e750.jpg',\n                'api': 'https://api.bibox.com',\n                'www': 'https://www.bibox.com',\n                'doc': 'https://github.com/Biboxcom/api_reference/wiki/home_en',\n                'fees': 'https://bibox.zendesk.com/hc/en-us/articles/115004417013-Fee-Structure-on-Bibox',\n            },\n            'api': {\n                'public': {\n                    'post': [\n                        // TODO: rework for full endpoint/cmd paths here\n                        'mdata',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'user',\n                        'orderpending',\n                        'transfer',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'taker': 0.001,\n                    'maker': 0.0,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                    },\n                    'deposit': 0.0,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicPostMdata ({\n            'cmd': 'api/marketAll',\n            'body': {},\n        });\n        let markets = response['result'];\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let base = market['coin_symbol'];\n            let quote = market['currency_symbol'];\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            let id = base + '_' + quote;\n            let precision = {\n                'amount': 8,\n                'price': 8,\n            };\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': undefined,\n                'info': market,\n                'lot': Math.pow (10, -precision['amount']),\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision['amount']),\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                },\n            }));\n        }\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.safeInteger (ticker, 'timestamp', this.seconds ());\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else {\n            symbol = ticker['coin_symbol'] + '/' + ticker['currency_symbol'];\n        }\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'buy'),\n            'ask': this.safeFloat (ticker, 'sell'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': this.safeString (ticker, 'percent'),\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'vol'),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicPostMdata ({\n            'cmd': 'api/ticker',\n            'body': this.extend ({\n                'pair': market['id'],\n            }, params),\n        });\n        return this.parseTicker (response['result'], market);\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        let response = await this.publicPostMdata ({\n            'cmd': 'api/marketAll',\n            'body': {},\n        });\n        let tickers = response['result'];\n        let result = {};\n        for (let t = 0; t < tickers.length; t++) {\n            let ticker = this.parseTicker (tickers[t]);\n            let symbol = ticker['symbol'];\n            if (symbols && (!(symbol in symbols))) {\n                continue;\n            }\n            result[symbol] = ticker;\n        }\n        return result;\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = trade['time'];\n        let side = (trade['side'] === '1') ? 'buy' : 'sell';\n        return {\n            'id': undefined,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': 'limit',\n            'side': side,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let size = (limit) ? limit : 200;\n        let response = await this.publicPostMdata ({\n            'cmd': 'api/deals',\n            'body': this.extend ({\n                'pair': market['id'],\n                'size': size,\n            }, params),\n        });\n        return this.parseTrades (response['result'], market, since, limit);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicPostMdata ({\n            'cmd': 'api/depth',\n            'body': this.extend ({\n                'pair': market['id'],\n            }, params),\n        });\n        return this.parseOrderBook (response['result'], this.safeFloat (response['result'], 'update_time'), 'bids', 'asks', 'price', 'amount');\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv['time'],\n            ohlcv['open'],\n            ohlcv['high'],\n            ohlcv['low'],\n            ohlcv['close'],\n            ohlcv['vol'],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let size = (limit) ? limit : 1000;\n        let response = await this.publicPostMdata ({\n            'cmd': 'api/kline',\n            'body': this.extend ({\n                'pair': market['id'],\n                'period': this.timeframes[timeframe],\n                'size': size,\n            }, params),\n        });\n        return this.parseOHLCVs (response['result'], market, timeframe, since, limit);\n    }\n\n    async fetchCurrencies (params = {}) {\n        let response = await this.privatePostTransfer ({\n            'cmd': 'transfer/coinList',\n            'body': {},\n        });\n        let currencies = response['result'];\n        let result = {};\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let id = currency['symbol'];\n            let code = this.commonCurrencyCode (id);\n            let precision = 8;\n            let deposit = currency['enable_deposit'];\n            let withdraw = currency['enable_withdraw'];\n            let active = (deposit && withdraw);\n            result[code] = {\n                'id': id,\n                'code': code,\n                'info': currency,\n                'name': currency['name'],\n                'active': active,\n                'status': 'ok',\n                'fee': undefined,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': undefined,\n                        'max': Math.pow (10, precision),\n                    },\n                },\n            };\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostTransfer ({\n            'cmd': 'transfer/assets',\n            'body': this.extend ({\n                'select': 1,\n            }, params),\n        });\n        let balances = response['result'];\n        let result = { 'info': balances };\n        let indexed = undefined;\n        if ('assets_list' in balances) {\n            indexed = this.indexBy (balances['assets_list'], 'coin_symbol');\n        } else {\n            indexed = {};\n        }\n        let keys = Object.keys (indexed);\n        for (let i = 0; i < keys.length; i++) {\n            let id = keys[i];\n            let currency = this.commonCurrencyCode (id);\n            let account = this.account ();\n            let balance = indexed[id];\n            let used = parseFloat (balance['freeze']);\n            let free = parseFloat (balance['balance']);\n            let total = this.sum (free, used);\n            account['free'] = free;\n            account['used'] = used;\n            account['total'] = total;\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orderType = (type === 'limit') ? 2 : 1;\n        let response = await this.privatePostOrder ({\n            'cmd': 'orderpending/trade',\n            'body': this.extend ({\n                'pair': market['id'],\n                'account_type': 0,\n                'order_type': orderType,\n                'order_side': side,\n                'pay_bix': 0,\n                'amount': amount,\n                'price': price,\n            }, params),\n        });\n        return {\n            'info': response,\n            'id': this.safeString (response, 'result'),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        let response = await this.privatePostCancelOrder ({\n            'cmd': 'orderpending/cancelTrade',\n            'body': this.extend ({\n                'orders_id': id,\n            }, params),\n        });\n        return response;\n    }\n\n    parseOrder (order, market = undefined) {\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else {\n            symbol = order['coin_symbol'] + '/' + order['currency_symbol'];\n        }\n        let type = (order['order_type'] === 1) ? 'market' : 'limit';\n        let timestamp = order['createdAt'];\n        let price = order['price'];\n        let filled = order['amount'];\n        let amount = this.safeInteger (order, 'deal_amount');\n        let remaining = amount - filled;\n        let side = (order['order_side'] === 1) ? 'buy' : 'sell';\n        let status = undefined;\n        if ('status' in order) {\n            status = this.parseOrderStatus (order['status']);\n        }\n        let result = {\n            'info': order,\n            'id': this.safeString (order, 'id'),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': price,\n            'amount': amount,\n            'cost': price * filled,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': this.safeFloat (order, 'fee'),\n        };\n        return result;\n    }\n\n    parseOrderStatus (status) {\n        let statuses = {\n            '1': 'pending',\n            '2': 'open',\n            '3': 'closed',\n            '4': 'canceled',\n            '5': 'canceled',\n            '6': 'canceled',\n        };\n        return this.safeString (statuses, status, status.toLowerCase ());\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrders requires a symbol param');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let size = (limit) ? limit : 200;\n        let response = await this.privatePostOrderpending ({\n            'cmd': 'orderpending/orderPendingList',\n            'body': this.extend ({\n                'pair': market['id'],\n                'account_type': 0, // 0 - regular, 1 - margin\n                'page': 1,\n                'size': size,\n            }, params),\n        });\n        let orders = ('items' in response) ? response['items'] : [];\n        return this.parseOrders (orders, market, since, limit);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchMyTrades requires a symbol param');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let size = (limit) ? limit : 200;\n        let response = await this.privatePostOrderpending ({\n            'cmd': 'orderpending/orderHistoryList',\n            'body': this.extend ({\n                'pair': market['id'],\n                'account_type': 0, // 0 - regular, 1 - margin\n                'page': 1,\n                'size': size,\n            }, params),\n        });\n        let orders = ('items' in response) ? response['items'] : [];\n        return this.parseOrders (orders, market, since, limit);\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (currency);\n        let response = await this.privatePostTransfer ({\n            'cmd': 'transfer/transferOutInfo',\n            'body': this.extend ({\n                'coin_symbol': market['id'],\n            }, params),\n        });\n        let result = {\n            'info': response,\n            'address': undefined,\n        };\n        return result;\n    }\n\n    async withdraw (code, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let currency = this.currency (code);\n        let response = await this.privatePostTransfer ({\n            'cmd': 'transfer/transferOut',\n            'body': this.extend ({\n                'coin_symbol': currency,\n                'amount': amount,\n                'addr': address,\n                'addr_remark': '',\n            }, params),\n        });\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + path;\n        let cmds = this.json ([ params ]);\n        if (api === 'public') {\n            body = {\n                'cmds': cmds,\n            };\n        } else {\n            this.checkRequiredCredentials ();\n            body = {\n                'cmds': cmds,\n                'apikey': this.apiKey,\n                'sign': this.hmac (this.encode (cmds), this.encode (this.secret), 'md5'),\n            };\n        }\n        headers = { 'Content-Type': 'application/json' };\n        return { 'url': url, 'method': method, 'body': this.json (body), 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        let message = this.id + ' ' + this.json (response);\n        if ('error' in response) {\n            if ('code' in response['error']) {\n                let code = response['error']['code'];\n                if (code === '3012')\n                    throw new AuthenticationError (message); // invalid api key\n                else if (code === '3025')\n                    throw new AuthenticationError (message); // signature failed\n                else if (code === '4003')\n                    throw new DDoSProtection (message); // server is busy, try again later\n            }\n            throw new ExchangeError (message);\n        }\n        if (!('result' in response))\n            throw new ExchangeError (message);\n        return response['result'][0];\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InsufficientFunds, OrderNotFound, InvalidOrder, DDoSProtection } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class binance extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'binance',\n            'name': 'Binance',\n            'countries': 'JP', // Japan\n            'rateLimit': 500,\n            'hasCORS': false,\n            // obsolete metainfo interface\n            'hasFetchBidsAsks': true,\n            'hasFetchTickers': true,\n            'hasFetchOHLCV': true,\n            'hasFetchMyTrades': true,\n            'hasFetchOrder': true,\n            'hasFetchOrders': true,\n            'hasFetchOpenOrders': true,\n            'hasWithdraw': true,\n            // new metainfo interface\n            'has': {\n                'fetchBidsAsks': true,\n                'fetchTickers': true,\n                'fetchOHLCV': true,\n                'fetchMyTrades': true,\n                'fetchOrder': true,\n                'fetchOrders': true,\n                'fetchOpenOrders': true,\n                'withdraw': true,\n            },\n            'timeframes': {\n                '1m': '1m',\n                '3m': '3m',\n                '5m': '5m',\n                '15m': '15m',\n                '30m': '30m',\n                '1h': '1h',\n                '2h': '2h',\n                '4h': '4h',\n                '6h': '6h',\n                '8h': '8h',\n                '12h': '12h',\n                '1d': '1d',\n                '3d': '3d',\n                '1w': '1w',\n                '1M': '1M',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/29604020-d5483cdc-87ee-11e7-94c7-d1a8d9169293.jpg',\n                'api': {\n                    'web': 'https://www.binance.com',\n                    'wapi': 'https://api.binance.com/wapi/v3',\n                    'public': 'https://api.binance.com/api/v1',\n                    'private': 'https://api.binance.com/api/v3',\n                    'v3': 'https://api.binance.com/api/v3',\n                    'v1': 'https://api.binance.com/api/v1',\n                },\n                'www': 'https://www.binance.com',\n                'doc': 'https://github.com/binance-exchange/binance-official-api-docs/blob/master/rest-api.md',\n                'fees': [\n                    'https://binance.zendesk.com/hc/en-us/articles/115000429332',\n                    'https://support.binance.com/hc/en-us/articles/115000583311',\n                ],\n            },\n            'api': {\n                'web': {\n                    'get': [\n                        'exchange/public/product',\n                    ],\n                },\n                'wapi': {\n                    'post': [\n                        'withdraw',\n                    ],\n                    'get': [\n                        'depositHistory',\n                        'withdrawHistory',\n                        'depositAddress',\n                    ],\n                },\n                'v3': {\n                    'get': [\n                        'ticker/price',\n                        'ticker/bookTicker',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'exchangeInfo',\n                        'ping',\n                        'time',\n                        'depth',\n                        'aggTrades',\n                        'klines',\n                        'ticker/24hr',\n                        'ticker/allPrices',\n                        'ticker/allBookTickers',\n                        'ticker/price',\n                        'ticker/bookTicker',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'order',\n                        'openOrders',\n                        'allOrders',\n                        'account',\n                        'myTrades',\n                    ],\n                    'post': [\n                        'order',\n                        'order/test',\n                    ],\n                    'delete': [\n                        'order',\n                    ],\n                },\n                'v1': {\n                    'put': [ 'userDataStream' ],\n                    'post': [ 'userDataStream' ],\n                    'delete': [ 'userDataStream' ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'taker': 0.001,\n                    'maker': 0.001,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BNB': 1.0,\n                        'BTC': 0.001,\n                        'ETH': 0.01,\n                        'LTC': 0.01,\n                        'NEO': 0.0,\n                        'QTUM': 0.01,\n                        'SNT': 10.0,\n                        'BNT': 1.2,\n                        'EOS': 0.7,\n                        'BCH': 0.0005,\n                        'GAS': 0.0,\n                        'USDT': 25.0,\n                        'OAX': 6.0,\n                        'DNT': 60.0,\n                        'MCO': 0.3,\n                        'ICN': 2.0,\n                        'WTC': 0.4,\n                        'OMG': 0.3,\n                        'ZRX': 10.0,\n                        'STRAT': 0.1,\n                        'SNGLS': 20.0,\n                        'BQX': 2.0,\n                        'KNC': 2.0,\n                        'FUN': 80.0,\n                        'SNM': 20.0,\n                        'LINK': 10.0,\n                        'XVG': 0.1,\n                        'CTR': 7.0,\n                        'SALT': 0.4,\n                        'IOTA': 0.5,\n                        'MDA': 2.0,\n                        'MTL': 0.5,\n                        'SUB': 4.0,\n                        'ETC': 0.01,\n                        'MTH': 35.0,\n                        'ENG': 5.0,\n                        'AST': 10.0,\n                        'BTG': undefined,\n                        'DASH': 0.002,\n                        'EVX': 2.5,\n                        'REQ': 15.0,\n                        'LRC': 12.0,\n                        'VIB': 20.0,\n                        'HSR': 0.0001,\n                        'TRX': 30.0,\n                        'POWR': 5.0,\n                        'ARK': 0.1,\n                        'YOYO': 10.0,\n                        'XRP': 0.15,\n                        'MOD': 2.0,\n                        'ENJ': 80.0,\n                        'STORJ': 3.0,\n                        'VEN': 5.0,\n                        'KMD': 1.0,\n                        'NULS': 4.0,\n                        'RCN': 20.0,\n                        'RDN': 0.3,\n                        'XMR': 0.04,\n                        'DLT': 15.0,\n                        'AMB': 10.0,\n                        'BAT': 15.0,\n                        'ZEC': 0.005,\n                        'BCPT': 14.0,\n                        'ARN': 7.0,\n                        'GVT': 0.5,\n                        'CDT': 35.0,\n                        'GXS': 0.3,\n                        'POE': 50.0,\n                        'QSP': 30.0,\n                        'BTS': 1.0,\n                        'XZC': 0.02,\n                        'LSK': 0.1,\n                        'TNT': 35.0,\n                        'FUEL': 60.0,\n                        'MANA': 30.0,\n                        'BCD': 0.0005,\n                        'DGD': 0.03,\n                        'ADX': 2.0,\n                        'ADA': 1.0,\n                        'PPT': 0.1,\n                        'CMT': 15.0,\n                        'XLM': 0.01,\n                        'CND': 180.0,\n                        'LEND': 50.0,\n                        'WABI': 4.0,\n                        'TNB': 70.0,\n                        'WAVES': 0.002,\n                        'ICX': 1.5,\n                        'GTO': 30.0,\n                        'OST': 15.0,\n                        'ELF': 2.0,\n                        'AION': 1.0,\n                        'NEBL': 0.01,\n                        'BRD': 3.0,\n                        'EDO': 1.5,\n                        'WINGS': 3.0,\n                        'NAV': 0.2,\n                        'LUN': 0.3,\n                        'TRIG': 5.0,\n                    },\n                    'deposit': {\n                        'BNB': 0,\n                        'BTC': 0,\n                        'ETH': 0,\n                        'LTC': 0,\n                        'NEO': 0,\n                        'QTUM': 0,\n                        'SNT': 0,\n                        'BNT': 0,\n                        'EOS': 0,\n                        'BCH': 0,\n                        'GAS': 0,\n                        'USDT': 0,\n                        'OAX': 0,\n                        'DNT': 0,\n                        'MCO': 0,\n                        'ICN': 0,\n                        'WTC': 0,\n                        'OMG': 0,\n                        'ZRX': 0,\n                        'STRAT': 0,\n                        'SNGLS': 0,\n                        'BQX': 0,\n                        'KNC': 0,\n                        'FUN': 0,\n                        'SNM': 0,\n                        'LINK': 0,\n                        'XVG': 0,\n                        'CTR': 0,\n                        'SALT': 0,\n                        'IOTA': 0,\n                        'MDA': 0,\n                        'MTL': 0,\n                        'SUB': 0,\n                        'ETC': 0,\n                        'MTH': 0,\n                        'ENG': 0,\n                        'AST': 0,\n                        'BTG': 0,\n                        'DASH': 0,\n                        'EVX': 0,\n                        'REQ': 0,\n                        'LRC': 0,\n                        'VIB': 0,\n                        'HSR': 0,\n                        'TRX': 0,\n                        'POWR': 0,\n                        'ARK': 0,\n                        'YOYO': 0,\n                        'XRP': 0,\n                        'MOD': 0,\n                        'ENJ': 0,\n                        'STORJ': 0,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicGetExchangeInfo ();\n        let markets = response['symbols'];\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let id = market['symbol'];\n            if (id === '123456')\n                continue;\n            let baseId = market['baseAsset'];\n            let quoteId = market['quoteAsset'];\n            let base = this.commonCurrencyCode (baseId);\n            let quote = this.commonCurrencyCode (quoteId);\n            let symbol = base + '/' + quote;\n            let filters = this.indexBy (market['filters'], 'filterType');\n            let precision = {\n                'base': market['baseAssetPrecision'],\n                'quote': market['quotePrecision'],\n                'amount': market['baseAssetPrecision'],\n                'price': market['quotePrecision'],\n            };\n            let active = (market['status'] === 'TRADING');\n            let lot = -1 * Math.log10 (precision['amount']);\n            let entry = this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'baseId': baseId,\n                'quoteId': quoteId,\n                'info': market,\n                'lot': lot,\n                'active': active,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': -1 * Math.log10 (precision['price']),\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': lot,\n                        'max': undefined,\n                    },\n                },\n            });\n            if ('PRICE_FILTER' in filters) {\n                let filter = filters['PRICE_FILTER'];\n                entry['precision']['price'] = this.precisionFromString (filter['tickSize']);\n                entry['limits']['price'] = {\n                    'min': parseFloat (filter['minPrice']),\n                    'max': parseFloat (filter['maxPrice']),\n                };\n            }\n            if ('LOT_SIZE' in filters) {\n                let filter = filters['LOT_SIZE'];\n                entry['precision']['amount'] = this.precisionFromString (filter['stepSize']);\n                entry['lot'] = parseFloat (filter['stepSize']);\n                entry['limits']['amount'] = {\n                    'min': parseFloat (filter['minQty']),\n                    'max': parseFloat (filter['maxQty']),\n                };\n            }\n            if ('MIN_NOTIONAL' in filters) {\n                entry['limits']['cost']['min'] = parseFloat (filters['MIN_NOTIONAL']['minNotional']);\n            }\n            result.push (entry);\n        }\n        return result;\n    }\n\n    calculateFee (symbol, type, side, amount, price, takerOrMaker = 'taker', params = {}) {\n        let market = this.markets[symbol];\n        let key = 'quote';\n        let rate = market[takerOrMaker];\n        let cost = parseFloat (this.costToPrecision (symbol, amount * rate));\n        if (side === 'sell') {\n            cost *= price;\n        } else {\n            key = 'base';\n        }\n        return {\n            'type': takerOrMaker,\n            'currency': market[key],\n            'rate': rate,\n            'cost': parseFloat (this.feeToPrecision (symbol, cost)),\n        };\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetAccount (params);\n        let result = { 'info': response };\n        let balances = response['balances'];\n        for (let i = 0; i < balances.length; i++) {\n            let balance = balances[i];\n            let asset = balance['asset'];\n            let currency = this.commonCurrencyCode (asset);\n            let account = {\n                'free': parseFloat (balance['free']),\n                'used': parseFloat (balance['locked']),\n                'total': 0.0,\n            };\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orderbook = await this.publicGetDepth (this.extend ({\n            'symbol': market['id'],\n            'limit': 100, // default = maximum = 100\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.safeInteger (ticker, 'closeTime');\n        if (typeof timestamp === 'undefined')\n            timestamp = this.milliseconds ();\n        let symbol = ticker['symbol'];\n        if (!market) {\n            if (symbol in this.markets_by_id) {\n                market = this.markets_by_id[symbol];\n            }\n        }\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'highPrice'),\n            'low': this.safeFloat (ticker, 'lowPrice'),\n            'bid': this.safeFloat (ticker, 'bidPrice'),\n            'bidVolume': this.safeFloat (ticker, 'bidQty'),\n            'ask': this.safeFloat (ticker, 'askPrice'),\n            'askVolume': this.safeFloat (ticker, 'askQty'),\n            'vwap': this.safeFloat (ticker, 'weightedAvgPrice'),\n            'open': this.safeFloat (ticker, 'openPrice'),\n            'close': this.safeFloat (ticker, 'prevClosePrice'),\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'lastPrice'),\n            'change': this.safeFloat (ticker, 'priceChangePercent'),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'volume'),\n            'quoteVolume': this.safeFloat (ticker, 'quoteVolume'),\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTicker24hr (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTicker (response, market);\n    }\n\n    parseTickers (rawTickers, symbols = undefined) {\n        let tickers = [];\n        for (let i = 0; i < rawTickers.length; i++) {\n            tickers.push (this.parseTicker (rawTickers[i]));\n        }\n        let tickersBySymbol = this.indexBy (tickers, 'symbol');\n        // return all of them if no symbols were passed in the first argument\n        if (typeof symbols === 'undefined')\n            return tickersBySymbol;\n        // otherwise filter by symbol\n        let result = {};\n        for (let i = 0; i < symbols.length; i++) {\n            let symbol = symbols[i];\n            if (symbol in tickersBySymbol)\n                result[symbol] = tickersBySymbol[symbol];\n        }\n        return result;\n    }\n\n    async fetchBidAsks (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let rawTickers = await this.publicGetTickerBookTicker (params);\n        return this.parseTickers (rawTickers, symbols);\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let rawTickers = await this.publicGetTicker24hr (params);\n        return this.parseTickers (rawTickers, symbols);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv[0],\n            parseFloat (ohlcv[1]),\n            parseFloat (ohlcv[2]),\n            parseFloat (ohlcv[3]),\n            parseFloat (ohlcv[4]),\n            parseFloat (ohlcv[5]),\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n            'interval': this.timeframes[timeframe],\n        };\n        request['limit'] = (limit) ? limit : 500; // default == max == 500\n        if (since)\n            request['startTime'] = since;\n        let response = await this.publicGetKlines (this.extend (request, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestampField = ('T' in trade) ? 'T' : 'time';\n        let timestamp = trade[timestampField];\n        let priceField = ('p' in trade) ? 'p' : 'price';\n        let price = parseFloat (trade[priceField]);\n        let amountField = ('q' in trade) ? 'q' : 'qty';\n        let amount = parseFloat (trade[amountField]);\n        let idField = ('a' in trade) ? 'a' : 'id';\n        let id = trade[idField].toString ();\n        let side = undefined;\n        let order = undefined;\n        if ('orderId' in trade)\n            order = trade['orderId'].toString ();\n        if ('m' in trade) {\n            side = trade['m'] ? 'sell' : 'buy'; // this is reversed intentionally\n        } else {\n            side = (trade['isBuyer']) ? 'buy' : 'sell'; // this is a true side\n        }\n        let fee = undefined;\n        if ('commission' in trade) {\n            fee = {\n                'cost': parseFloat (trade['commission']),\n                'currency': this.commonCurrencyCode (trade['commissionAsset']),\n            };\n        }\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': id,\n            'order': order,\n            'type': undefined,\n            'side': side,\n            'price': price,\n            'cost': price * amount,\n            'amount': amount,\n            'fee': fee,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n        };\n        if (since) {\n            request['startTime'] = since;\n            request['endTime'] = since + 3600000;\n        }\n        if (limit)\n            request['limit'] = limit;\n        // 'fromId': 123,    // ID to get aggregate trades from INCLUSIVE.\n        // 'startTime': 456, // Timestamp in ms to get aggregate trades from INCLUSIVE.\n        // 'endTime': 789,   // Timestamp in ms to get aggregate trades until INCLUSIVE.\n        // 'limit': 500,     // default = maximum = 500\n        let response = await this.publicGetAggTrades (this.extend (request, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    parseOrderStatus (status) {\n        if (status === 'NEW')\n            return 'open';\n        if (status === 'PARTIALLY_FILLED')\n            return 'open';\n        if (status === 'FILLED')\n            return 'closed';\n        if (status === 'CANCELED')\n            return 'canceled';\n        return status.toLowerCase ();\n    }\n\n    parseOrder (order, market = undefined) {\n        let status = this.parseOrderStatus (order['status']);\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else {\n            let id = order['symbol'];\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            }\n        }\n        let timestamp = undefined;\n        if ('time' in order)\n            timestamp = order['time'];\n        else if ('transactTime' in order)\n            timestamp = order['transactTime'];\n        else\n            throw new ExchangeError (this.id + ' malformed order: ' + this.json (order));\n        let price = parseFloat (order['price']);\n        let amount = parseFloat (order['origQty']);\n        let filled = this.safeFloat (order, 'executedQty', 0.0);\n        let remaining = Math.max (amount - filled, 0.0);\n        let result = {\n            'info': order,\n            'id': order['orderId'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': order['type'].toLowerCase (),\n            'side': order['side'].toLowerCase (),\n            'price': price,\n            'amount': amount,\n            'cost': price * amount,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': undefined,\n        };\n        return result;\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let order = {\n            'symbol': market['id'],\n            'quantity': this.amountToString (symbol, amount),\n            'type': type.toUpperCase (),\n            'side': side.toUpperCase (),\n        };\n        if (type === 'limit') {\n            order = this.extend (order, {\n                'price': this.priceToPrecision (symbol, price),\n                'timeInForce': 'GTC', // 'GTC' = Good To Cancel (default), 'IOC' = Immediate Or Cancel\n            });\n        }\n        let response = await this.privatePostOrder (this.extend (order, params));\n        return this.parseOrder (response);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrder requires a symbol param');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.privateGetOrder (this.extend ({\n            'symbol': market['id'],\n            'orderId': parseInt (id),\n        }, params));\n        return this.parseOrder (response, market);\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrders requires a symbol param');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n        };\n        if (limit)\n            request['limit'] = limit;\n        let response = await this.privateGetAllOrders (this.extend (request, params));\n        return this.parseOrders (response, market, since, limit);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOpenOrders requires a symbol param');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.privateGetOpenOrders (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseOrders (response, market, since, limit);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, since, limit, params);\n        return this.filterBy (orders, 'status', 'closed');\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' cancelOrder requires a symbol argument');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = undefined;\n        try {\n            response = await this.privateDeleteOrder (this.extend ({\n                'symbol': market['id'],\n                'orderId': parseInt (id),\n                // 'origClientOrderId': id,\n            }, params));\n        } catch (e) {\n            if (this.last_http_response.indexOf ('UNKNOWN_ORDER') >= 0)\n                throw new OrderNotFound (this.id + ' cancelOrder() error: ' + this.last_http_response);\n            throw e;\n        }\n        return response;\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchMyTrades requires a symbol argument');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n        };\n        if (limit)\n            request['limit'] = limit;\n        let response = await this.privateGetMyTrades (this.extend (request, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    commonCurrencyCode (currency) {\n        if (currency === 'BCC')\n            return 'BCH';\n        return currency;\n    }\n\n    currencyId (currency) {\n        if (currency === 'BCH')\n            return 'BCC';\n        return currency;\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        let response = await this.wapiGetDepositAddress (this.extend ({\n            'asset': this.currencyId (currency),\n        }, params));\n        if ('success' in response) {\n            if (response['success']) {\n                let address = this.safeString (response, 'address');\n                return {\n                    'currency': currency,\n                    'address': address,\n                    'status': 'ok',\n                    'info': response,\n                };\n            }\n        }\n        throw new ExchangeError (this.id + ' fetchDepositAddress failed: ' + this.last_http_response);\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        let request = {\n            'asset': this.currencyId (currency),\n            'address': address,\n            'amount': parseFloat (amount),\n            'name': address,\n        };\n        if (tag)\n            request['addressTag'] = tag;\n        let response = await this.wapiPostWithdraw (this.extend (request, params));\n        return {\n            'info': response,\n            'id': this.safeString (response, 'id'),\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        url += '/' + path;\n        if (api === 'wapi')\n            url += '.html';\n        // v1 special case for userDataStream\n        if (path === 'userDataStream') {\n            body = this.urlencode (params);\n            headers = {\n                'X-MBX-APIKEY': this.apiKey,\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        } else if ((api === 'private') || (api === 'wapi')) {\n            this.checkRequiredCredentials ();\n            let nonce = this.milliseconds ();\n            let query = this.urlencode (this.extend ({\n                'timestamp': nonce,\n                'recvWindow': 100000,\n            }, params));\n            let signature = this.hmac (this.encode (query), this.encode (this.secret));\n            query += '&' + 'signature=' + signature;\n            headers = {\n                'X-MBX-APIKEY': this.apiKey,\n            };\n            if ((method === 'GET') || (api === 'wapi')) {\n                url += '?' + query;\n            } else {\n                body = query;\n                headers['Content-Type'] = 'application/x-www-form-urlencoded';\n            }\n        } else {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code >= 400) {\n            if (code === 418)\n                throw new DDoSProtection (this.id + ' ' + code.toString () + ' ' + reason + ' ' + body);\n            if (body.indexOf ('Price * QTY is zero or less') >= 0)\n                throw new InvalidOrder (this.id + ' order cost = amount * price is zero or less ' + body);\n            if (body.indexOf ('MIN_NOTIONAL') >= 0)\n                throw new InvalidOrder (this.id + ' order cost = amount * price is too small ' + body);\n            if (body.indexOf ('LOT_SIZE') >= 0)\n                throw new InvalidOrder (this.id + ' order amount should be evenly divisible by lot size, use this.amountToLots (symbol, amount) ' + body);\n            if (body.indexOf ('PRICE_FILTER') >= 0)\n                throw new InvalidOrder (this.id + ' order price exceeds allowed price precision or invalid, use this.priceToPrecision (symbol, amount) ' + body);\n            if (body.indexOf ('Order does not exist') >= 0)\n                throw new OrderNotFound (this.id + ' ' + body);\n        }\n        if (body[0] === '{') {\n            let response = JSON.parse (body);\n            let error = this.safeValue (response, 'code');\n            if (typeof error !== 'undefined') {\n                if (error === -2010) {\n                    throw new InsufficientFunds (this.id + ' ' + this.json (response));\n                } else if (error === -2011) {\n                    throw new OrderNotFound (this.id + ' ' + this.json (response));\n                } else if (error === -1013) { // Invalid quantity\n                    throw new InvalidOrder (this.id + ' ' + this.json (response));\n                }\n            }\n        }\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange')\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bit2c extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bit2c',\n            'name': 'Bit2C',\n            'countries': 'IL', // Israel\n            'rateLimit': 3000,\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766119-3593220e-5ece-11e7-8b3a-5a041f6bcc3f.jpg',\n                'api': 'https://www.bit2c.co.il',\n                'www': 'https://www.bit2c.co.il',\n                'doc': [\n                    'https://www.bit2c.co.il/home/api',\n                    'https://github.com/OferE/bit2c',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'Exchanges/{pair}/Ticker',\n                        'Exchanges/{pair}/orderbook',\n                        'Exchanges/{pair}/trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'Account/Balance',\n                        'Account/Balance/v2',\n                        'Merchant/CreateCheckout',\n                        'Order/AccountHistory',\n                        'Order/AddCoinFundsRequest',\n                        'Order/AddFund',\n                        'Order/AddOrder',\n                        'Order/AddOrderMarketPriceBuy',\n                        'Order/AddOrderMarketPriceSell',\n                        'Order/CancelOrder',\n                        'Order/MyOrders',\n                        'Payment/GetMyId',\n                        'Payment/Send',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/NIS': { 'id': 'BtcNis', 'symbol': 'BTC/NIS', 'base': 'BTC', 'quote': 'NIS' },\n                'BCH/NIS': { 'id': 'BchNis', 'symbol': 'BCH/NIS', 'base': 'BCH', 'quote': 'NIS' },\n                'LTC/NIS': { 'id': 'LtcNis', 'symbol': 'LTC/NIS', 'base': 'LTC', 'quote': 'NIS' },\n                'BTG/NIS': { 'id': 'BtgNis', 'symbol': 'BTG/NIS', 'base': 'BTG', 'quote': 'NIS' },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.5 / 100,\n                    'taker': 0.5 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let balance = await this.privatePostAccountBalanceV2 ();\n        let result = { 'info': balance };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            if (currency in balance) {\n                let available = 'AVAILABLE_' + currency;\n                account['free'] = balance[available];\n                account['total'] = balance[currency];\n                account['used'] = account['total'] - account['free'];\n            }\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetExchangesPairOrderbook (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetExchangesPairTicker (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        let timestamp = this.milliseconds ();\n        let averagePrice = parseFloat (ticker['av']);\n        let baseVolume = parseFloat (ticker['a']);\n        let quoteVolume = baseVolume * averagePrice;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['h']),\n            'ask': parseFloat (ticker['l']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['ll']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': averagePrice,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'id': trade['tid'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'order': undefined,\n            'type': undefined,\n            'side': undefined,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetExchangesPairTrades (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let method = 'privatePostOrderAddOrder';\n        let order = {\n            'Amount': amount,\n            'Pair': this.marketId (symbol),\n        };\n        if (type === 'market') {\n            method += 'MarketPrice' + this.capitalize (side);\n        } else {\n            order['Price'] = price;\n            order['Total'] = amount * price;\n            order['IsBid'] = (side === 'buy');\n        }\n        let result = await this[method] (this.extend (order, params));\n        return {\n            'info': result,\n            'id': result['NewOrder']['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostOrderCancelOrder ({ 'id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        if (api === 'public') {\n            url += '.json';\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let query = this.extend ({ 'nonce': nonce }, params);\n            body = this.urlencode (query);\n            let signature = this.hmac (this.encode (body), this.encode (this.secret), 'sha512', 'base64');\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'key': this.apiKey,\n                'sign': this.decode (signature),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { InvalidNonce, InsufficientFunds, AuthenticationError, InvalidOrder, ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitbay extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitbay',\n            'name': 'BitBay',\n            'countries': [ 'PL', 'EU' ], // Poland\n            'rateLimit': 1000,\n            'hasCORS': true,\n            'hasWithdraw': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766132-978a7bd8-5ece-11e7-9540-bc96d1e9bbb8.jpg',\n                'www': 'https://bitbay.net',\n                'api': {\n                    'public': 'https://bitbay.net/API/Public',\n                    'private': 'https://bitbay.net/API/Trading/tradingApi.php',\n                },\n                'doc': [\n                    'https://bitbay.net/public-api',\n                    'https://bitbay.net/account/tab-api',\n                    'https://github.com/BitBayNet/API',\n                ],\n                'fees': 'https://bitbay.net/en/fees',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        '{id}/all',\n                        '{id}/market',\n                        '{id}/orderbook',\n                        '{id}/ticker',\n                        '{id}/trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'info',\n                        'trade',\n                        'cancel',\n                        'orderbook',\n                        'orders',\n                        'transfer',\n                        'withdraw',\n                        'history',\n                        'transactions',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/USD': { 'id': 'BTCUSD', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD', 'baseId': 'BTC', 'quoteId': 'USD' },\n                'BTC/EUR': { 'id': 'BTCEUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR', 'baseId': 'BTC', 'quoteId': 'EUR' },\n                'BTC/PLN': { 'id': 'BTCPLN', 'symbol': 'BTC/PLN', 'base': 'BTC', 'quote': 'PLN', 'baseId': 'BTC', 'quoteId': 'PLN' },\n                'LTC/USD': { 'id': 'LTCUSD', 'symbol': 'LTC/USD', 'base': 'LTC', 'quote': 'USD', 'baseId': 'LTC', 'quoteId': 'USD' },\n                'LTC/EUR': { 'id': 'LTCEUR', 'symbol': 'LTC/EUR', 'base': 'LTC', 'quote': 'EUR', 'baseId': 'LTC', 'quoteId': 'EUR' },\n                'LTC/PLN': { 'id': 'LTCPLN', 'symbol': 'LTC/PLN', 'base': 'LTC', 'quote': 'PLN', 'baseId': 'LTC', 'quoteId': 'PLN' },\n                'LTC/BTC': { 'id': 'LTCBTC', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC', 'baseId': 'LTC', 'quoteId': 'BTC' },\n                'ETH/USD': { 'id': 'ETHUSD', 'symbol': 'ETH/USD', 'base': 'ETH', 'quote': 'USD', 'baseId': 'ETH', 'quoteId': 'USD' },\n                'ETH/EUR': { 'id': 'ETHEUR', 'symbol': 'ETH/EUR', 'base': 'ETH', 'quote': 'EUR', 'baseId': 'ETH', 'quoteId': 'EUR' },\n                'ETH/PLN': { 'id': 'ETHPLN', 'symbol': 'ETH/PLN', 'base': 'ETH', 'quote': 'PLN', 'baseId': 'ETH', 'quoteId': 'PLN' },\n                'ETH/BTC': { 'id': 'ETHBTC', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC', 'baseId': 'ETH', 'quoteId': 'BTC' },\n                'LSK/USD': { 'id': 'LSKUSD', 'symbol': 'LSK/USD', 'base': 'LSK', 'quote': 'USD', 'baseId': 'LSK', 'quoteId': 'USD' },\n                'LSK/EUR': { 'id': 'LSKEUR', 'symbol': 'LSK/EUR', 'base': 'LSK', 'quote': 'EUR', 'baseId': 'LSK', 'quoteId': 'EUR' },\n                'LSK/PLN': { 'id': 'LSKPLN', 'symbol': 'LSK/PLN', 'base': 'LSK', 'quote': 'PLN', 'baseId': 'LSK', 'quoteId': 'PLN' },\n                'LSK/BTC': { 'id': 'LSKBTC', 'symbol': 'LSK/BTC', 'base': 'LSK', 'quote': 'BTC', 'baseId': 'LSK', 'quoteId': 'BTC' },\n                'BCH/USD': { 'id': 'BCCUSD', 'symbol': 'BCH/USD', 'base': 'BCH', 'quote': 'USD', 'baseId': 'BCC', 'quoteId': 'USD' },\n                'BCH/EUR': { 'id': 'BCCEUR', 'symbol': 'BCH/EUR', 'base': 'BCH', 'quote': 'EUR', 'baseId': 'BCC', 'quoteId': 'EUR' },\n                'BCH/PLN': { 'id': 'BCCPLN', 'symbol': 'BCH/PLN', 'base': 'BCH', 'quote': 'PLN', 'baseId': 'BCC', 'quoteId': 'PLN' },\n                'BCH/BTC': { 'id': 'BCCBTC', 'symbol': 'BCH/BTC', 'base': 'BCH', 'quote': 'BTC', 'baseId': 'BCC', 'quoteId': 'BTC' },\n                'BTG/USD': { 'id': 'BTGUSD', 'symbol': 'BTG/USD', 'base': 'BTG', 'quote': 'USD', 'baseId': 'BTG', 'quoteId': 'USD' },\n                'BTG/EUR': { 'id': 'BTGEUR', 'symbol': 'BTG/EUR', 'base': 'BTG', 'quote': 'EUR', 'baseId': 'BTG', 'quoteId': 'EUR' },\n                'BTG/PLN': { 'id': 'BTGPLN', 'symbol': 'BTG/PLN', 'base': 'BTG', 'quote': 'PLN', 'baseId': 'BTG', 'quoteId': 'PLN' },\n                'BTG/BTC': { 'id': 'BTGBTC', 'symbol': 'BTG/BTC', 'base': 'BTG', 'quote': 'BTC', 'baseId': 'BTG', 'quoteId': 'BTC' },\n                'DASH/USD': { 'id': 'DASHUSD', 'symbol': 'DASH/USD', 'base': 'DASH', 'quote': 'USD', 'baseId': 'DASH', 'quoteId': 'USD' },\n                'DASH/EUR': { 'id': 'DASHEUR', 'symbol': 'DASH/EUR', 'base': 'DASH', 'quote': 'EUR', 'baseId': 'DASH', 'quoteId': 'EUR' },\n                'DASH/PLN': { 'id': 'DASHPLN', 'symbol': 'DASH/PLN', 'base': 'DASH', 'quote': 'PLN', 'baseId': 'DASH', 'quoteId': 'PLN' },\n                'DASH/BTC': { 'id': 'DASHBTC', 'symbol': 'DASH/BTC', 'base': 'DASH', 'quote': 'BTC', 'baseId': 'DASH', 'quoteId': 'BTC' },\n                'GAME/USD': { 'id': 'GAMEUSD', 'symbol': 'GAME/USD', 'base': 'GAME', 'quote': 'USD', 'baseId': 'GAME', 'quoteId': 'USD' },\n                'GAME/EUR': { 'id': 'GAMEEUR', 'symbol': 'GAME/EUR', 'base': 'GAME', 'quote': 'EUR', 'baseId': 'GAME', 'quoteId': 'EUR' },\n                'GAME/PLN': { 'id': 'GAMEPLN', 'symbol': 'GAME/PLN', 'base': 'GAME', 'quote': 'PLN', 'baseId': 'GAME', 'quoteId': 'PLN' },\n                'GAME/BTC': { 'id': 'GAMEBTC', 'symbol': 'GAME/BTC', 'base': 'GAME', 'quote': 'BTC', 'baseId': 'GAME', 'quoteId': 'BTC' },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.3 / 100,\n                    'taker': 0.0043,\n                },\n                'funding': {\n                    'withdraw': {\n                        'BTC': 0.0009,\n                        'LTC': 0.005,\n                        'ETH': 0.00126,\n                        'LSK': 0.2,\n                        'BCH': 0.0006,\n                        'GAME': 0.005,\n                        'DASH': 0.001,\n                        'BTG': 0.0008,\n                        'PLN': 4,\n                        'EUR': 1.5,\n                    },\n                },\n            },\n            'exceptions': {\n                '400': ExchangeError, // At least one parameter wasn't set\n                '401': InvalidOrder, // Invalid order type\n                '402': InvalidOrder, // No orders with specified currencies\n                '403': InvalidOrder, // Invalid payment currency name\n                '404': InvalidOrder, // Error. Wrong transaction type\n                '405': InvalidOrder, // Order with this id doesn't exist\n                '406': InsufficientFunds, // No enough money or crypto\n                // code 407 not specified are not specified in their docs\n                '408': InvalidOrder, // Invalid currency name\n                '501': AuthenticationError, // Invalid public key\n                '502': AuthenticationError, // Invalid sign\n                '503': InvalidNonce, // Invalid moment parameter. Request time doesn't match current server time\n                '504': ExchangeError, // Invalid method\n                '505': AuthenticationError, // Key has no permission for this action\n                '506': AuthenticationError, // Account locked. Please contact with customer service\n                // codes 507 and 508 are not specified in their docs\n                '509': ExchangeError, // The BIC/SWIFT is required for this currency\n                '510': ExchangeError, // Invalid market name\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostInfo ();\n        if ('balances' in response) {\n            let balance = response['balances'];\n            let result = { 'info': balance };\n            let codes = Object.keys (this.currencies);\n            for (let i = 0; i < codes.length; i++) {\n                let code = codes[i];\n                let currency = this.currencies[code];\n                let id = currency['id'];\n                let account = this.account ();\n                if (id in balance) {\n                    account['free'] = parseFloat (balance[id]['available']);\n                    account['used'] = parseFloat (balance[id]['locked']);\n                    account['total'] = this.sum (account['free'], account['used']);\n                }\n                result[code] = account;\n            }\n            return this.parseBalance (result);\n        }\n        throw new ExchangeError (this.id + ' empty balance response ' + this.json (response));\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetIdOrderbook (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetIdTicker (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        let timestamp = this.milliseconds ();\n        let baseVolume = this.safeFloat (ticker, 'volume');\n        let vwap = this.safeFloat (ticker, 'vwap');\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'max'),\n            'low': this.safeFloat (ticker, 'min'),\n            'bid': this.safeFloat (ticker, 'bid'),\n            'ask': this.safeFloat (ticker, 'ask'),\n            'vwap': vwap,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': this.safeFloat (ticker, 'average'),\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['date'] * 1000;\n        return {\n            'id': trade['tid'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['type'],\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetIdTrades (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let market = this.market (symbol);\n        return this.privatePostTrade (this.extend ({\n            'type': side,\n            'currency': market['baseId'],\n            'amount': amount,\n            'payment_currency': market['quoteId'],\n            'rate': price,\n        }, params));\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancel ({ 'id': id });\n    }\n\n    isFiat (currency) {\n        let fiatCurrencies = {\n            'USD': true,\n            'EUR': true,\n            'PLN': true,\n        };\n        if (currency in fiatCurrencies)\n            return true;\n        return false;\n    }\n\n    async withdraw (code, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let method = undefined;\n        let currency = this.currency (code);\n        let request = {\n            'currency': currency['id'],\n            'quantity': amount,\n        };\n        if (this.isFiat (code)) {\n            method = 'privatePostWithdraw';\n            // request['account'] = params['account']; // they demand an account number\n            // request['express'] = params['express']; // whatever it means, they don't explain\n            // request['bic'] = '';\n        } else {\n            method = 'privatePostTransfer';\n            request['address'] = address;\n        }\n        let response = await this[method] (this.extend (request, params));\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        if (api === 'public') {\n            url += '/' + this.implodeParams (path, params) + '.json';\n        } else {\n            this.checkRequiredCredentials ();\n            body = this.urlencode (this.extend ({\n                'method': path,\n                'moment': this.nonce (),\n            }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'API-Key': this.apiKey,\n                'API-Hash': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (httpCode, reason, url, method, headers, body) {\n        if ((typeof body !== 'string') || (body.length < 2))\n            return; // fallback to default error handler\n        if ((body[0] === '{') || (body[0] === '[')) {\n            let response = JSON.parse (body);\n            if ('code' in response) {\n                //\n                // bitbay returns the integer 'success': 1 key from their private API\n                // or an integer 'code' value from 0 to 510 and an error message\n                //\n                //      { 'success': 1, ... }\n                //      { 'code': 502, 'message': 'Invalid sign' }\n                //      { 'code': 0, 'message': 'offer funds not exceeding minimums' }\n                //\n                let code = response['code']; // always an integer\n                const feedback = this.id + ' ' + this.json (response);\n                const exceptions = this.exceptions;\n                if (code in this.exceptions) {\n                    throw new exceptions[code] (feedback);\n                } else {\n                    throw new ExchangeError (feedback);\n                }\n            }\n        }\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitcoincoid extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitcoincoid',\n            'name': 'Bitcoin.co.id',\n            'countries': 'ID', // Indonesia\n            'hasCORS': false,\n            // obsolete metainfo interface\n            'hasFetchTickers': false,\n            'hasFetchOHLCV': false,\n            'hasFetchOrder': true,\n            'hasFetchOrders': false,\n            'hasFetchClosedOrders': true,\n            'hasFetchOpenOrders': true,\n            'hasFetchMyTrades': false,\n            'hasFetchCurrencies': false,\n            'hasWithdraw': false,\n            // new metainfo interface\n            'has': {\n                'fetchTickers': false,\n                'fetchOHLCV': false,\n                'fetchOrder': true,\n                'fetchOrders': false,\n                'fetchClosedOrders': true,\n                'fetchOpenOrders': true,\n                'fetchMyTrades': false,\n                'fetchCurrencies': false,\n                'withdraw': false,\n            },\n            'version': '1.7', // as of 6 November 2017\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766138-043c7786-5ecf-11e7-882b-809c14f38b53.jpg',\n                'api': {\n                    'public': 'https://vip.bitcoin.co.id/api',\n                    'private': 'https://vip.bitcoin.co.id/tapi',\n                },\n                'www': 'https://www.bitcoin.co.id',\n                'doc': [\n                    'https://vip.bitcoin.co.id/downloads/BITCOINCOID-API-DOCUMENTATION.pdf',\n                    'https://vip.bitcoin.co.id/trade_api',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        '{pair}/ticker',\n                        '{pair}/trades',\n                        '{pair}/depth',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'getInfo',\n                        'transHistory',\n                        'trade',\n                        'tradeHistory',\n                        'getOrder',\n                        'openOrders',\n                        'cancelOrder',\n                        'orderHistory',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/IDR': { 'id': 'btc_idr', 'symbol': 'BTC/IDR', 'base': 'BTC', 'quote': 'IDR', 'baseId': 'btc', 'quoteId': 'idr' },\n                'BCH/IDR': { 'id': 'bch_idr', 'symbol': 'BCH/IDR', 'base': 'BCH', 'quote': 'IDR', 'baseId': 'bch', 'quoteId': 'idr' },\n                'BTG/IDR': { 'id': 'btg_idr', 'symbol': 'BTG/IDR', 'base': 'BTG', 'quote': 'IDR', 'baseId': 'btg', 'quoteId': 'idr' },\n                'ETH/IDR': { 'id': 'eth_idr', 'symbol': 'ETH/IDR', 'base': 'ETH', 'quote': 'IDR', 'baseId': 'eth', 'quoteId': 'idr' },\n                'ETC/IDR': { 'id': 'etc_idr', 'symbol': 'ETC/IDR', 'base': 'ETC', 'quote': 'IDR', 'baseId': 'etc', 'quoteId': 'idr' },\n                'IGNIS/IDR': { 'id': 'ignis_idr', 'symbol': 'IGNIS/IDR', 'base': 'IGNIS', 'quote': 'IDR', 'baseId': 'ignis', 'quoteId': 'idr' },\n                'LTC/IDR': { 'id': 'ltc_idr', 'symbol': 'LTC/IDR', 'base': 'LTC', 'quote': 'IDR', 'baseId': 'ltc', 'quoteId': 'idr' },\n                'NXT/IDR': { 'id': 'nxt_idr', 'symbol': 'NXT/IDR', 'base': 'NXT', 'quote': 'IDR', 'baseId': 'nxt', 'quoteId': 'idr' },\n                'WAVES/IDR': { 'id': 'waves_idr', 'symbol': 'WAVES/IDR', 'base': 'WAVES', 'quote': 'IDR', 'baseId': 'waves', 'quoteId': 'idr' },\n                'XRP/IDR': { 'id': 'xrp_idr', 'symbol': 'XRP/IDR', 'base': 'XRP', 'quote': 'IDR', 'baseId': 'xrp', 'quoteId': 'idr' },\n                'XZC/IDR': { 'id': 'xzc_idr', 'symbol': 'XZC/IDR', 'base': 'XZC', 'quote': 'IDR', 'baseId': 'xzc', 'quoteId': 'idr' },\n                'XLM/IDR': { 'id': 'str_idr', 'symbol': 'XLM/IDR', 'base': 'XLM', 'quote': 'IDR', 'baseId': 'str', 'quoteId': 'idr' },\n                'BTS/BTC': { 'id': 'bts_btc', 'symbol': 'BTS/BTC', 'base': 'BTS', 'quote': 'BTC', 'baseId': 'bts', 'quoteId': 'btc' },\n                'DASH/BTC': { 'id': 'drk_btc', 'symbol': 'DASH/BTC', 'base': 'DASH', 'quote': 'BTC', 'baseId': 'drk', 'quoteId': 'btc' },\n                'DOGE/BTC': { 'id': 'doge_btc', 'symbol': 'DOGE/BTC', 'base': 'DOGE', 'quote': 'BTC', 'baseId': 'doge', 'quoteId': 'btc' },\n                'ETH/BTC': { 'id': 'eth_btc', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC', 'baseId': 'eth', 'quoteId': 'btc' },\n                'LTC/BTC': { 'id': 'ltc_btc', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC', 'baseId': 'ltc', 'quoteId': 'btc' },\n                'NXT/BTC': { 'id': 'nxt_btc', 'symbol': 'NXT/BTC', 'base': 'NXT', 'quote': 'BTC', 'baseId': 'nxt', 'quoteId': 'btc' },\n                'XLM/BTC': { 'id': 'str_btc', 'symbol': 'XLM/BTC', 'base': 'XLM', 'quote': 'BTC', 'baseId': 'str', 'quoteId': 'btc' },\n                'XEM/BTC': { 'id': 'nem_btc', 'symbol': 'XEM/BTC', 'base': 'XEM', 'quote': 'BTC', 'baseId': 'nem', 'quoteId': 'btc' },\n                'XRP/BTC': { 'id': 'xrp_btc', 'symbol': 'XRP/BTC', 'base': 'XRP', 'quote': 'BTC', 'baseId': 'xrp', 'quoteId': 'btc' },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetInfo ();\n        let balance = response['return'];\n        let result = { 'info': balance };\n        let codes = Object.keys (this.currencies);\n        for (let i = 0; i < codes.length; i++) {\n            let code = codes[i];\n            let currency = this.currencies[code];\n            let lowercase = currency['id'];\n            let account = this.account ();\n            account['free'] = this.safeFloat (balance['balance'], lowercase, 0.0);\n            account['used'] = this.safeFloat (balance['balance_hold'], lowercase, 0.0);\n            account['total'] = this.sum (account['free'], account['used']);\n            result[code] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetPairDepth (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'buy', 'sell');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetPairTicker (this.extend ({\n            'pair': market['id'],\n        }, params));\n        let ticker = response['ticker'];\n        let timestamp = parseFloat (ticker['server_time']) * 1000;\n        let baseVolume = 'vol_' + market['baseId'].toLowerCase ();\n        let quoteVolume = 'vol_' + market['quoteId'].toLowerCase ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy']),\n            'ask': parseFloat (ticker['sell']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker[baseVolume]),\n            'quoteVolume': parseFloat (ticker[quoteVolume]),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        return {\n            'id': trade['tid'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['type'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetPairTrades (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    parseOrder (order, market = undefined) {\n        let side = undefined;\n        if ('type' in order)\n            side = order['type'];\n        let status = this.safeString (order, 'status', 'open');\n        if (status == 'filled') {\n            status = 'closed';\n        } else if (status == 'calcelled') {\n            status = 'canceled';\n        }\n        let symbol = undefined;\n        let cost = undefined;\n        let price = this.safeFloat (order, 'price');\n        let amount = undefined;\n        let remaining = undefined;\n        let filled = undefined;\n        if (market) {\n            symbol = market['symbol'];\n            let quoteId = market['quoteId'];\n            let baseId = market['baseId'];\n            if ((market['quoteId'] == 'idr') && ('order_rp' in order))\n                quoteId = 'rp';\n            if ((market['baseId'] == 'idr') && ('remain_rp' in order))\n                baseId = 'rp';\n            cost = this.safeFloat (order, 'order_' + quoteId);\n            if (cost) {\n                amount = cost / price;\n                let remainingCost = this.safeFloat (order, 'remain_' + quoteId);\n                if (typeof remainingCost !== 'undefined') {\n                    remaining = remainingCost / price;\n                    filled = amount - remaining;\n                }\n            } else {\n                amount = this.safeFloat (order, 'order_' + baseId);\n                cost = price * amount;\n                remaining = this.safeFloat (order, 'remain_' + baseId);\n                filled = amount - remaining;\n            }\n        }\n        let average = undefined;\n        if (filled)\n            average = cost / filled;\n        let timestamp = parseInt (order['submit_time']);\n        let fee = undefined;\n        let result = {\n            'info': order,\n            'id': order['order_id'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': 'limit',\n            'side': side,\n            'price': price,\n            'cost': cost,\n            'average': average,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': fee,\n        };\n        return result;\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrder requires a symbol');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.privatePostGetOrder (this.extend ({\n            'pair': market['id'],\n            'order_id': id,\n        }, params));\n        let orders = response['return'];\n        let order = this.parseOrder (this.extend ({ 'id': id }, orders['order']), market);\n        return this.extend ({ 'info': response }, order);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOpenOrders requires a symbol');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'pair': market['id'],\n        };\n        let response = await this.privatePostOpenOrders (this.extend (request, params));\n        let orders = this.parseOrders (response['return']['orders'], market, since, limit);\n        return this.filterOrdersBySymbol (orders, symbol);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrders requires a symbol');\n        await this.loadMarkets ();\n        let request = {};\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['pair'] = market['id'];\n        }\n        let response = await this.privatePostOrderHistory (this.extend (request, params));\n        let orders = this.parseOrders (response['return']['orders'], market, since, limit);\n        orders = this.filterBy (orders, 'status', 'closed');\n        if (symbol)\n            return this.filterOrdersBySymbol (orders, symbol);\n        return orders;\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let order = {\n            'pair': market['id'],\n            'type': side,\n            'price': price,\n        };\n        let base = market['baseId'];\n        order[base] = amount;\n        let result = await this.privatePostTrade (this.extend (order, params));\n        return {\n            'info': result,\n            'id': result['return']['order_id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelOrder (this.extend ({\n            'order_id': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        if (api == 'public') {\n            url += '/' + this.implodeParams (path, params);\n        } else {\n            this.checkRequiredCredentials ();\n            body = this.urlencode (this.extend ({\n                'method': path,\n                'nonce': this.nonce (),\n            }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Key': this.apiKey,\n                'Sign': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response)\n            throw new ExchangeError (this.id + ' ' + response['error']);\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { DDoSProtection, AuthenticationError, ExchangeError, InsufficientFunds, NotSupported, InvalidOrder, OrderNotFound } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitfinex extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitfinex',\n            'name': 'Bitfinex',\n            'countries': 'VG',\n            'version': 'v1',\n            'rateLimit': 1500,\n            'hasCORS': false,\n            // old metainfo interface\n            'hasFetchOrder': true,\n            'hasFetchTickers': true,\n            'hasDeposit': true,\n            'hasWithdraw': true,\n            'hasFetchOHLCV': true,\n            'hasFetchOpenOrders': true,\n            'hasFetchClosedOrders': true,\n            // new metainfo interface\n            'has': {\n                'fetchOHLCV': true,\n                'fetchTickers': true,\n                'fetchOrder': true,\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': true,\n                'fetchMyTrades': true,\n                'withdraw': true,\n                'deposit': true,\n            },\n            'timeframes': {\n                '1m': '1m',\n                '5m': '5m',\n                '15m': '15m',\n                '30m': '30m',\n                '1h': '1h',\n                '3h': '3h',\n                '6h': '6h',\n                '12h': '12h',\n                '1d': '1D',\n                '1w': '7D',\n                '2w': '14D',\n                '1M': '1M',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766244-e328a50c-5ed2-11e7-947b-041416579bb3.jpg',\n                'api': 'https://api.bitfinex.com',\n                'www': 'https://www.bitfinex.com',\n                'doc': [\n                    'https://bitfinex.readme.io/v1/docs',\n                    'https://github.com/bitfinexcom/bitfinex-api-node',\n                ],\n            },\n            'api': {\n                'v2': {\n                    'get': [\n                        'candles/trade:{timeframe}:{symbol}/{section}',\n                        'candles/trade:{timeframe}:{symbol}/last',\n                        'candles/trade:{timeframe}:{symbol}/hist',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'book/{symbol}',\n                        // 'candles/{symbol}',\n                        'lendbook/{currency}',\n                        'lends/{currency}',\n                        'pubticker/{symbol}',\n                        'stats/{symbol}',\n                        'symbols',\n                        'symbols_details',\n                        'tickers',\n                        'today',\n                        'trades/{symbol}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'account_fees',\n                        'account_infos',\n                        'balances',\n                        'basket_manage',\n                        'credits',\n                        'deposit/new',\n                        'funding/close',\n                        'history',\n                        'history/movements',\n                        'key_info',\n                        'margin_infos',\n                        'mytrades',\n                        'mytrades_funding',\n                        'offer/cancel',\n                        'offer/new',\n                        'offer/status',\n                        'offers',\n                        'offers/hist',\n                        'order/cancel',\n                        'order/cancel/all',\n                        'order/cancel/multi',\n                        'order/cancel/replace',\n                        'order/new',\n                        'order/new/multi',\n                        'order/status',\n                        'orders',\n                        'orders/hist',\n                        'position/claim',\n                        'positions',\n                        'summary',\n                        'taken_funds',\n                        'total_taken_funds',\n                        'transfer',\n                        'unused_taken_funds',\n                        'withdraw',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': true,\n                    'percentage': true,\n                    'maker': 0.1 / 100,\n                    'taker': 0.2 / 100,\n                    'tiers': {\n                        'taker': [\n                            [0, 0.2 / 100],\n                            [500000, 0.2 / 100],\n                            [1000000, 0.2 / 100],\n                            [2500000, 0.2 / 100],\n                            [5000000, 0.2 / 100],\n                            [7500000, 0.2 / 100],\n                            [10000000, 0.18 / 100],\n                            [15000000, 0.16 / 100],\n                            [20000000, 0.14 / 100],\n                            [25000000, 0.12 / 100],\n                            [30000000, 0.1 / 100],\n                        ],\n                        'maker': [\n                            [0, 0.1 / 100],\n                            [500000, 0.08 / 100],\n                            [1000000, 0.06 / 100],\n                            [2500000, 0.04 / 100],\n                            [5000000, 0.02 / 100],\n                            [7500000, 0],\n                            [10000000, 0],\n                            [15000000, 0],\n                            [20000000, 0],\n                            [25000000, 0],\n                            [30000000, 0],\n                        ],\n                    },\n                },\n                'funding': {\n                    'tierBased': false, // true for tier-based/progressive\n                    'percentage': false, // fixed commission\n                    'deposit': {\n                        'BTC': 0.0005,\n                        'IOTA': 0.5,\n                        'ETH': 0.01,\n                        'BCH': 0.01,\n                        'LTC': 0.1,\n                        'EOS': 0.1,\n                        'XMR': 0.04,\n                        'SAN': 0.1,\n                        'DASH': 0.01,\n                        'ETC': 0.01,\n                        'XPR': 0.02,\n                        'YYW': 0.1,\n                        'NEO': 0,\n                        'ZEC': 0.1,\n                        'BTG': 0,\n                        'OMG': 0.1,\n                        'DATA': 1,\n                        'QASH': 1,\n                        'ETP': 0.01,\n                        'QTUM': 0.01,\n                        'EDO': 0.5,\n                        'AVT': 0.5,\n                        'USDT': 0,\n                    },\n                    'withdraw': {\n                        'BTC': 0.0005,\n                        'IOTA': 0.5,\n                        'ETH': 0.01,\n                        'BCH': 0.01,\n                        'LTC': 0.1,\n                        'EOS': 0.1,\n                        'XMR': 0.04,\n                        'SAN': 0.1,\n                        'DASH': 0.01,\n                        'ETC': 0.01,\n                        'XPR': 0.02,\n                        'YYW': 0.1,\n                        'NEO': 0,\n                        'ZEC': 0.1,\n                        'BTG': 0,\n                        'OMG': 0.1,\n                        'DATA': 1,\n                        'QASH': 1,\n                        'ETP': 0.01,\n                        'QTUM': 0.01,\n                        'EDO': 0.5,\n                        'AVT': 0.5,\n                        'USDT': 5,\n                    },\n                },\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        const currencies = {\n            'DSH': 'DASH', // Bitfinex names Dash as DSH, instead of DASH\n            'QTM': 'QTUM',\n            'BCC': 'CST_BCC',\n            'BCU': 'CST_BCU',\n            'IOT': 'IOTA',\n            'DAT': 'DATA',\n        };\n        return (currency in currencies) ? currencies[currency] : currency;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetSymbolsDetails ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['pair'].toUpperCase ();\n            let baseId = id.slice (0, 3);\n            let quoteId = id.slice (3, 6);\n            let base = this.commonCurrencyCode (baseId);\n            let quote = this.commonCurrencyCode (quoteId);\n            let symbol = base + '/' + quote;\n            let precision = {\n                'price': market['price_precision'],\n                'amount': market['price_precision'],\n            };\n            let limits = {\n                'amount': {\n                    'min': parseFloat (market['minimum_order_size']),\n                    'max': parseFloat (market['maximum_order_size']),\n                },\n                'price': {\n                    'min': Math.pow (10, -precision['price']),\n                    'max': Math.pow (10, precision['price']),\n                },\n            };\n            limits['cost'] = {\n                'min': limits['amount']['min'] * limits['price']['min'],\n                'max': undefined,\n            };\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'baseId': baseId,\n                'quoteId': quoteId,\n                'active': true,\n                'precision': precision,\n                'limits': limits,\n                'lot': Math.pow (10, -precision['amount']),\n                'info': market,\n            }));\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balanceType = this.safeString (params, 'type', 'exchange');\n        let balances = await this.privatePostBalances ();\n        let result = { 'info': balances };\n        for (let i = 0; i < balances.length; i++) {\n            let balance = balances[i];\n            if (balance['type'] === balanceType) {\n                let currency = balance['currency'];\n                let uppercase = currency.toUpperCase ();\n                uppercase = this.commonCurrencyCode (uppercase);\n                let account = this.account ();\n                account['free'] = parseFloat (balance['available']);\n                account['total'] = parseFloat (balance['amount']);\n                account['used'] = account['total'] - account['free'];\n                result[uppercase] = account;\n            }\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetBookSymbol (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'amount');\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTickers (params);\n        let result = {};\n        for (let i = 0; i < tickers.length; i++) {\n            let ticker = tickers[i];\n            if ('pair' in ticker) {\n                let id = ticker['pair'];\n                if (id in this.markets_by_id) {\n                    let market = this.markets_by_id[id];\n                    let symbol = market['symbol'];\n                    result[symbol] = this.parseTicker (ticker, market);\n                } else {\n                    throw new ExchangeError (this.id + ' fetchTickers() failed to recognize symbol ' + id + ' ' + this.json (ticker));\n                }\n            } else {\n                throw new ExchangeError (this.id + ' fetchTickers() response not recognized ' + this.json (tickers));\n            }\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetPubtickerSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = parseFloat (ticker['timestamp']) * 1000;\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else if ('pair' in ticker) {\n            let id = ticker['pair'];\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            } else {\n                throw new ExchangeError (this.id + ' unrecognized ticker symbol ' + id + ' ' + this.json (ticker));\n            }\n        }\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last_price']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': parseFloat (ticker['mid']),\n            'baseVolume': parseFloat (ticker['volume']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (parseFloat (trade['timestamp'])) * 1000;\n        let side = trade['type'].toLowerCase ();\n        let orderId = this.safeString (trade, 'order_id');\n        let price = parseFloat (trade['price']);\n        let amount = parseFloat (trade['amount']);\n        let cost = price * amount;\n        return {\n            'id': trade['tid'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'order': orderId,\n            'side': side,\n            'price': price,\n            'amount': amount,\n            'cost': cost,\n            'fee': undefined,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTradesSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = { 'symbol': market['id'] };\n        if (limit) {\n            request['limit_trades'] = limit;\n        }\n        if (since) {\n            request['timestamp'] = parseInt (since / 1000);\n        }\n        let response = await this.privatePostMytrades (this.extend (request, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let orderType = type;\n        if ((type === 'limit') || (type === 'market'))\n            orderType = 'exchange ' + type;\n        // amount = this.amountToPrecision (symbol, amount);\n        let order = {\n            'symbol': this.marketId (symbol),\n            'amount': amount.toString (),\n            'side': side,\n            'type': orderType,\n            'ocoorder': false,\n            'buy_price_oco': 0,\n            'sell_price_oco': 0,\n        };\n        if (type === 'market') {\n            order['price'] = this.nonce ().toString ();\n        } else {\n            // price = this.priceToPrecision (symbol, price);\n            order['price'] = price.toString ();\n        }\n        let result = await this.privatePostOrderNew (this.extend (order, params));\n        return this.parseOrder (result);\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostOrderCancel ({ 'order_id': parseInt (id) });\n    }\n\n    parseOrder (order, market = undefined) {\n        let side = order['side'];\n        let open = order['is_live'];\n        let canceled = order['is_cancelled'];\n        let status = undefined;\n        if (open) {\n            status = 'open';\n        } else if (canceled) {\n            status = 'canceled';\n        } else {\n            status = 'closed';\n        }\n        let symbol = undefined;\n        if (!market) {\n            let exchange = order['symbol'].toUpperCase ();\n            if (exchange in this.markets_by_id) {\n                market = this.markets_by_id[exchange];\n            }\n        }\n        if (market)\n            symbol = market['symbol'];\n        let orderType = order['type'];\n        let exchange = orderType.indexOf ('exchange ') >= 0;\n        if (exchange) {\n            let parts = order['type'].split (' ');\n            orderType = parts[1];\n        }\n        let timestamp = parseInt (parseFloat (order['timestamp']) * 1000);\n        let result = {\n            'info': order,\n            'id': order['id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': orderType,\n            'side': side,\n            'price': this.safeFloat (order, 'price'),\n            'average': parseFloat (order['avg_execution_price']),\n            'amount': parseFloat (order['original_amount']),\n            'remaining': parseFloat (order['remaining_amount']),\n            'filled': parseFloat (order['executed_amount']),\n            'status': status,\n            'fee': undefined,\n        };\n        return result;\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostOrders (params);\n        let orders = this.parseOrders (response, undefined, since, limit);\n        if (symbol)\n            orders = this.filterBy (orders, 'symbol', symbol);\n        return orders;\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        if (limit)\n            request['limit'] = limit;\n        let response = await this.privatePostOrdersHist (this.extend (request, params));\n        let orders = this.parseOrders (response, undefined, since, limit);\n        if (symbol)\n            orders = this.filterBy (orders, 'symbol', symbol);\n        orders = this.filterBy (orders, 'status', 'closed');\n        return orders;\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostOrderStatus (this.extend ({\n            'order_id': parseInt (id),\n        }, params));\n        return this.parseOrder (response);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv[0],\n            ohlcv[1],\n            ohlcv[3],\n            ohlcv[4],\n            ohlcv[2],\n            ohlcv[5],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let v2id = 't' + market['id'];\n        let request = {\n            'symbol': v2id,\n            'timeframe': this.timeframes[timeframe],\n            'sort': 1,\n        };\n        if (limit)\n            request['limit'] = limit;\n        if (since)\n            request['start'] = since;\n        request = this.extend (request, params);\n        let response = await this.v2GetCandlesTradeTimeframeSymbolHist (request);\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    getCurrencyName (currency) {\n        if (currency === 'BTC') {\n            return 'bitcoin';\n        } else if (currency === 'LTC') {\n            return 'litecoin';\n        } else if (currency === 'ETH') {\n            return 'ethereum';\n        } else if (currency === 'ETC') {\n            return 'ethereumc';\n        } else if (currency === 'OMNI') {\n            return 'mastercoin'; // ???\n        } else if (currency === 'ZEC') {\n            return 'zcash';\n        } else if (currency === 'XMR') {\n            return 'monero';\n        } else if (currency === 'USD') {\n            return 'wire';\n        } else if (currency === 'DASH') {\n            return 'dash';\n        } else if (currency === 'XRP') {\n            return 'ripple';\n        } else if (currency === 'EOS') {\n            return 'eos';\n        } else if (currency === 'BCH') {\n            return 'bcash';\n        } else if (currency === 'USDT') {\n            return 'tetheruso';\n        }\n        throw new NotSupported (this.id + ' ' + currency + ' not supported for withdrawal');\n    }\n\n    async createDepositAddress (currency, params = {}) {\n        let response = await this.fetchDepositAddress (currency, this.extend ({\n            'renew': 1,\n        }, params));\n        return {\n            'currency': currency,\n            'address': response['address'],\n            'status': 'ok',\n            'info': response['info'],\n        };\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        let name = this.getCurrencyName (currency);\n        let request = {\n            'method': name,\n            'wallet_name': 'exchange',\n            'renew': 0, // a value of 1 will generate a new address\n        };\n        let response = await this.privatePostDepositNew (this.extend (request, params));\n        return {\n            'currency': currency,\n            'address': response['address'],\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        let name = this.getCurrencyName (currency);\n        let request = {\n            'withdraw_type': name,\n            'walletselected': 'exchange',\n            'amount': amount.toString (),\n            'address': address,\n        };\n        if (tag)\n            request['payment_id'] = tag;\n        let responses = await this.privatePostWithdraw (this.extend (request, params));\n        let response = responses[0];\n        return {\n            'info': response,\n            'id': response['withdrawal_id'],\n        };\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let request = '/' + this.implodeParams (path, params);\n        if (api === 'v2') {\n            request = '/' + api + request;\n        } else {\n            request = '/' + this.version + request;\n        }\n        let query = this.omit (params, this.extractParams (path));\n        let url = this.urls['api'] + request;\n        if ((api === 'public') || (path.indexOf ('/hist') >= 0)) {\n            if (Object.keys (query).length) {\n                let suffix = '?' + this.urlencode (query);\n                url += suffix;\n                request += suffix;\n            }\n        }\n        if (api === 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            query = this.extend ({\n                'nonce': nonce.toString (),\n                'request': request,\n            }, query);\n            query = this.json (query);\n            query = this.encode (query);\n            let payload = this.stringToBase64 (query);\n            let secret = this.encode (this.secret);\n            let signature = this.hmac (payload, secret, 'sha384');\n            headers = {\n                'X-BFX-APIKEY': this.apiKey,\n                'X-BFX-PAYLOAD': this.decode (payload),\n                'X-BFX-SIGNATURE': signature,\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (body.length < 2)\n            return;\n        if (code >= 400) {\n            if (body[0] === '{') {\n                let response = JSON.parse (body);\n                if ('message' in response) {\n                    let message = response['message'];\n                    let error = this.id + ' ' + message;\n                    if (message.indexOf ('Key price should be a decimal number') >= 0) {\n                        throw new InvalidOrder (error);\n                    } else if (message.indexOf ('Invalid order: not enough exchange balance') >= 0) {\n                        throw new InsufficientFunds (error);\n                    } else if (message === 'Order could not be cancelled.') {\n                        throw new OrderNotFound (error);\n                    } else if (message.indexOf ('Invalid order') >= 0) {\n                        throw new InvalidOrder (error);\n                    } else if (message === 'Order price must be positive.') {\n                        throw new InvalidOrder (error);\n                    } else if (message.indexOf ('Key amount should be a decimal number') >= 0) {\n                        throw new InvalidOrder (error);\n                    } else if (message === 'No such order found.') {\n                        throw new OrderNotFound (error);\n                    } else if (message === 'Could not find a key matching the given X-BFX-APIKEY.') {\n                        throw new AuthenticationError (error);\n                    }\n                } else if ('error' in response) {\n                    let code = response['error'];\n                    let error = this.id + ' ' + code;\n                    if (code === 'ERR_RATE_LIMIT')\n                        throw new DDoSProtection (error);\n                }\n            }\n        }\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('message' in response) {\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        return response;\n    }\n};\n","'use strict';\n\n// ---------------------------------------------------------------------------\n\nconst bitfinex = require ('./bitfinex.js');\nconst { ExchangeError, NotSupported, InsufficientFunds } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class bitfinex2 extends bitfinex {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitfinex2',\n            'name': 'Bitfinex v2',\n            'countries': 'VG',\n            'version': 'v2',\n            'hasCORS': true,\n            // old metainfo interface\n            'hasCreateOrder': false,\n            'hasFetchOrder': true,\n            'hasFetchTickers': true,\n            'hasFetchOHLCV': true,\n            'hasWithdraw': true,\n            'hasDeposit': false,\n            'hasFetchOpenOrders': false,\n            'hasFetchClosedOrders': false,\n            // new metainfo interface\n            'has': {\n                'createOrder': false,\n                'fetchOHLCV': true,\n                'fetchTickers': true,\n                'fetchOrder': true,\n                'fetchOpenOrders': false,\n                'fetchClosedOrders': false,\n                'withdraw': true,\n                'deposit': false,\n            },\n            'timeframes': {\n                '1m': '1m',\n                '5m': '5m',\n                '15m': '15m',\n                '30m': '30m',\n                '1h': '1h',\n                '3h': '3h',\n                '6h': '6h',\n                '12h': '12h',\n                '1d': '1D',\n                '1w': '7D',\n                '2w': '14D',\n                '1M': '1M',\n            },\n            'rateLimit': 1500,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766244-e328a50c-5ed2-11e7-947b-041416579bb3.jpg',\n                'api': 'https://api.bitfinex.com',\n                'www': 'https://www.bitfinex.com',\n                'doc': [\n                    'https://bitfinex.readme.io/v2/docs',\n                    'https://github.com/bitfinexcom/bitfinex-api-node',\n                ],\n                'fees': 'https://www.bitfinex.com/fees',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'platform/status',\n                        'tickers',\n                        'ticker/{symbol}',\n                        'trades/{symbol}/hist',\n                        'book/{symbol}/{precision}',\n                        'book/{symbol}/P0',\n                        'book/{symbol}/P1',\n                        'book/{symbol}/P2',\n                        'book/{symbol}/P3',\n                        'book/{symbol}/R0',\n                        'symbols_details',\n                        'stats1/{key}:{size}:{symbol}/{side}/{section}',\n                        'stats1/{key}:{size}:{symbol}/long/last',\n                        'stats1/{key}:{size}:{symbol}/long/hist',\n                        'stats1/{key}:{size}:{symbol}/short/last',\n                        'stats1/{key}:{size}:{symbol}/short/hist',\n                        'candles/trade:{timeframe}:{symbol}/{section}',\n                        'candles/trade:{timeframe}:{symbol}/last',\n                        'candles/trade:{timeframe}:{symbol}/hist',\n                    ],\n                    'post': [\n                        'calc/trade/avg',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'auth/r/wallets',\n                        'auth/r/orders/{symbol}',\n                        'auth/r/orders/{symbol}/new',\n                        'auth/r/orders/{symbol}/hist',\n                        'auth/r/order/{symbol}:{id}/trades',\n                        'auth/r/trades/{symbol}/hist',\n                        'auth/r/positions',\n                        'auth/r/funding/offers/{symbol}',\n                        'auth/r/funding/offers/{symbol}/hist',\n                        'auth/r/funding/loans/{symbol}',\n                        'auth/r/funding/loans/{symbol}/hist',\n                        'auth/r/funding/credits/{symbol}',\n                        'auth/r/funding/credits/{symbol}/hist',\n                        'auth/r/funding/trades/{symbol}/hist',\n                        'auth/r/info/margin/{key}',\n                        'auth/r/info/funding/{key}',\n                        'auth/r/movements/{currency}/hist',\n                        'auth/r/stats/perf:{timeframe}/hist',\n                        'auth/r/alerts',\n                        'auth/w/alert/set',\n                        'auth/w/alert/{type}:{symbol}:{price}/del',\n                        'auth/calc/order/avail',\n                    ],\n                },\n            },\n            'markets': {\n                'AVT/BTC': { 'id': 'tAVTBTC', 'symbol': 'AVT/BTC', 'base': 'AVT', 'quote': 'BTC', 'baseId': 'tAVT', 'quoteId': 'tBTC' },\n                'AVT/ETH': { 'id': 'tAVTETH', 'symbol': 'AVT/ETH', 'base': 'AVT', 'quote': 'ETH', 'baseId': 'tAVT', 'quoteId': 'tETH' },\n                'AVT/USD': { 'id': 'tAVTUSD', 'symbol': 'AVT/USD', 'base': 'AVT', 'quote': 'USD', 'baseId': 'tAVT', 'quoteId': 'zUSD' },\n                'CST_BCC/BTC': { 'id': 'tBCCBTC', 'symbol': 'CST_BCC/BTC', 'base': 'CST_BCC', 'quote': 'BTC', 'baseId': 'tBCC', 'quoteId': 'tBTC' },\n                'CST_BCC/USD': { 'id': 'tBCCUSD', 'symbol': 'CST_BCC/USD', 'base': 'CST_BCC', 'quote': 'USD', 'baseId': 'tBCC', 'quoteId': 'zUSD' },\n                'BCH/BTC': { 'id': 'tBCHBTC', 'symbol': 'BCH/BTC', 'base': 'BCH', 'quote': 'BTC', 'baseId': 'tBCH', 'quoteId': 'tBTC' },\n                'BCH/ETH': { 'id': 'tBCHETH', 'symbol': 'BCH/ETH', 'base': 'BCH', 'quote': 'ETH', 'baseId': 'tBCH', 'quoteId': 'tETH' },\n                'BCH/USD': { 'id': 'tBCHUSD', 'symbol': 'BCH/USD', 'base': 'BCH', 'quote': 'USD', 'baseId': 'tBCH', 'quoteId': 'zUSD' },\n                'CST_BCU/BTC': { 'id': 'tBCUBTC', 'symbol': 'CST_BCU/BTC', 'base': 'CST_BCU', 'quote': 'BTC', 'baseId': 'tBCU', 'quoteId': 'tBTC' },\n                'CST_BCU/USD': { 'id': 'tBCUUSD', 'symbol': 'CST_BCU/USD', 'base': 'CST_BCU', 'quote': 'USD', 'baseId': 'tBCU', 'quoteId': 'zUSD' },\n                'BT1/BTC': { 'id': 'tBT1BTC', 'symbol': 'BT1/BTC', 'base': 'BT1', 'quote': 'BTC', 'baseId': 'tBT1', 'quoteId': 'tBTC' },\n                'BT1/USD': { 'id': 'tBT1USD', 'symbol': 'BT1/USD', 'base': 'BT1', 'quote': 'USD', 'baseId': 'tBT1', 'quoteId': 'zUSD' },\n                'BT2/BTC': { 'id': 'tBT2BTC', 'symbol': 'BT2/BTC', 'base': 'BT2', 'quote': 'BTC', 'baseId': 'tBT2', 'quoteId': 'tBTC' },\n                'BT2/USD': { 'id': 'tBT2USD', 'symbol': 'BT2/USD', 'base': 'BT2', 'quote': 'USD', 'baseId': 'tBT2', 'quoteId': 'zUSD' },\n                'BTC/USD': { 'id': 'tBTCUSD', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD', 'baseId': 'tBTC', 'quoteId': 'zUSD' },\n                'BTC/EUR': { 'id': 'tBTCEUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR', 'baseId': 'tBTC', 'quoteId': 'zEUR' },\n                'BTG/BTC': { 'id': 'tBTGBTC', 'symbol': 'BTG/BTC', 'base': 'BTG', 'quote': 'BTC', 'baseId': 'tBTG', 'quoteId': 'tBTC' },\n                'BTG/USD': { 'id': 'tBTGUSD', 'symbol': 'BTG/USD', 'base': 'BTG', 'quote': 'USD', 'baseId': 'tBTG', 'quoteId': 'zUSD' },\n                'DASH/BTC': { 'id': 'tDSHBTC', 'symbol': 'DASH/BTC', 'base': 'DASH', 'quote': 'BTC', 'baseId': 'tDASH', 'quoteId': 'tBTC' },\n                'DASH/USD': { 'id': 'tDSHUSD', 'symbol': 'DASH/USD', 'base': 'DASH', 'quote': 'USD', 'baseId': 'tDASH', 'quoteId': 'zUSD' },\n                'DAT/BTC': { 'id': 'tDATBTC', 'symbol': 'DAT/BTC', 'base': 'DAT', 'quote': 'BTC', 'baseId': 'tDAT', 'quoteId': 'tBTC' },\n                'DAT/ETH': { 'id': 'tDATETH', 'symbol': 'DAT/ETH', 'base': 'DAT', 'quote': 'ETH', 'baseId': 'tDAT', 'quoteId': 'tETH' },\n                'DAT/USD': { 'id': 'tDATUSD', 'symbol': 'DAT/USD', 'base': 'DAT', 'quote': 'USD', 'baseId': 'tDAT', 'quoteId': 'zUSD' },\n                'EDO/BTC': { 'id': 'tEDOBTC', 'symbol': 'EDO/BTC', 'base': 'EDO', 'quote': 'BTC', 'baseId': 'tEDO', 'quoteId': 'tBTC' },\n                'EDO/ETH': { 'id': 'tEDOETH', 'symbol': 'EDO/ETH', 'base': 'EDO', 'quote': 'ETH', 'baseId': 'tEDO', 'quoteId': 'tETH' },\n                'EDO/USD': { 'id': 'tEDOUSD', 'symbol': 'EDO/USD', 'base': 'EDO', 'quote': 'USD', 'baseId': 'tEDO', 'quoteId': 'zUSD' },\n                'EOS/BTC': { 'id': 'tEOSBTC', 'symbol': 'EOS/BTC', 'base': 'EOS', 'quote': 'BTC', 'baseId': 'tEOS', 'quoteId': 'tBTC' },\n                'EOS/ETH': { 'id': 'tEOSETH', 'symbol': 'EOS/ETH', 'base': 'EOS', 'quote': 'ETH', 'baseId': 'tEOS', 'quoteId': 'tETH' },\n                'EOS/USD': { 'id': 'tEOSUSD', 'symbol': 'EOS/USD', 'base': 'EOS', 'quote': 'USD', 'baseId': 'tEOS', 'quoteId': 'zUSD' },\n                'ETC/BTC': { 'id': 'tETCBTC', 'symbol': 'ETC/BTC', 'base': 'ETC', 'quote': 'BTC', 'baseId': 'tETC', 'quoteId': 'tBTC' },\n                'ETC/USD': { 'id': 'tETCUSD', 'symbol': 'ETC/USD', 'base': 'ETC', 'quote': 'USD', 'baseId': 'tETC', 'quoteId': 'zUSD' },\n                'ETH/BTC': { 'id': 'tETHBTC', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC', 'baseId': 'tETH', 'quoteId': 'tBTC' },\n                'ETH/USD': { 'id': 'tETHUSD', 'symbol': 'ETH/USD', 'base': 'ETH', 'quote': 'USD', 'baseId': 'tETH', 'quoteId': 'zUSD' },\n                'ETP/BTC': { 'id': 'tETPBTC', 'symbol': 'ETP/BTC', 'base': 'ETP', 'quote': 'BTC', 'baseId': 'tETP', 'quoteId': 'tBTC' },\n                'ETP/ETH': { 'id': 'tETPETH', 'symbol': 'ETP/ETH', 'base': 'ETP', 'quote': 'ETH', 'baseId': 'tETP', 'quoteId': 'tETH' },\n                'ETP/USD': { 'id': 'tETPUSD', 'symbol': 'ETP/USD', 'base': 'ETP', 'quote': 'USD', 'baseId': 'tETP', 'quoteId': 'zUSD' },\n                'IOTA/BTC': { 'id': 'tIOTBTC', 'symbol': 'IOTA/BTC', 'base': 'IOTA', 'quote': 'BTC', 'baseId': 'tIOTA', 'quoteId': 'tBTC' },\n                'IOTA/ETH': { 'id': 'tIOTETH', 'symbol': 'IOTA/ETH', 'base': 'IOTA', 'quote': 'ETH', 'baseId': 'tIOTA', 'quoteId': 'tETH' },\n                'IOTA/USD': { 'id': 'tIOTUSD', 'symbol': 'IOTA/USD', 'base': 'IOTA', 'quote': 'USD', 'baseId': 'tIOTA', 'quoteId': 'zUSD' },\n                'LTC/BTC': { 'id': 'tLTCBTC', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC', 'baseId': 'tLTC', 'quoteId': 'tBTC' },\n                'LTC/USD': { 'id': 'tLTCUSD', 'symbol': 'LTC/USD', 'base': 'LTC', 'quote': 'USD', 'baseId': 'tLTC', 'quoteId': 'zUSD' },\n                'NEO/BTC': { 'id': 'tNEOBTC', 'symbol': 'NEO/BTC', 'base': 'NEO', 'quote': 'BTC', 'baseId': 'tNEO', 'quoteId': 'tBTC' },\n                'NEO/ETH': { 'id': 'tNEOETH', 'symbol': 'NEO/ETH', 'base': 'NEO', 'quote': 'ETH', 'baseId': 'tNEO', 'quoteId': 'tETH' },\n                'NEO/USD': { 'id': 'tNEOUSD', 'symbol': 'NEO/USD', 'base': 'NEO', 'quote': 'USD', 'baseId': 'tNEO', 'quoteId': 'zUSD' },\n                'OMG/BTC': { 'id': 'tOMGBTC', 'symbol': 'OMG/BTC', 'base': 'OMG', 'quote': 'BTC', 'baseId': 'tOMG', 'quoteId': 'tBTC' },\n                'OMG/ETH': { 'id': 'tOMGETH', 'symbol': 'OMG/ETH', 'base': 'OMG', 'quote': 'ETH', 'baseId': 'tOMG', 'quoteId': 'tETH' },\n                'OMG/USD': { 'id': 'tOMGUSD', 'symbol': 'OMG/USD', 'base': 'OMG', 'quote': 'USD', 'baseId': 'tOMG', 'quoteId': 'zUSD' },\n                'QTUM/BTC': { 'id': 'tQTMBTC', 'symbol': 'QTUM/BTC', 'base': 'QTUM', 'quote': 'BTC', 'baseId': 'tQTUM', 'quoteId': 'tBTC' },\n                'QTUM/ETH': { 'id': 'tQTMETH', 'symbol': 'QTUM/ETH', 'base': 'QTUM', 'quote': 'ETH', 'baseId': 'tQTUM', 'quoteId': 'tETH' },\n                'QTUM/USD': { 'id': 'tQTMUSD', 'symbol': 'QTUM/USD', 'base': 'QTUM', 'quote': 'USD', 'baseId': 'tQTUM', 'quoteId': 'zUSD' },\n                'RRT/BTC': { 'id': 'tRRTBTC', 'symbol': 'RRT/BTC', 'base': 'RRT', 'quote': 'BTC', 'baseId': 'tRRT', 'quoteId': 'tBTC' },\n                'RRT/USD': { 'id': 'tRRTUSD', 'symbol': 'RRT/USD', 'base': 'RRT', 'quote': 'USD', 'baseId': 'tRRT', 'quoteId': 'zUSD' },\n                'SAN/BTC': { 'id': 'tSANBTC', 'symbol': 'SAN/BTC', 'base': 'SAN', 'quote': 'BTC', 'baseId': 'tSAN', 'quoteId': 'tBTC' },\n                'SAN/ETH': { 'id': 'tSANETH', 'symbol': 'SAN/ETH', 'base': 'SAN', 'quote': 'ETH', 'baseId': 'tSAN', 'quoteId': 'tETH' },\n                'SAN/USD': { 'id': 'tSANUSD', 'symbol': 'SAN/USD', 'base': 'SAN', 'quote': 'USD', 'baseId': 'tSAN', 'quoteId': 'zUSD' },\n                'XMR/BTC': { 'id': 'tXMRBTC', 'symbol': 'XMR/BTC', 'base': 'XMR', 'quote': 'BTC', 'baseId': 'tXMR', 'quoteId': 'tBTC' },\n                'XMR/USD': { 'id': 'tXMRUSD', 'symbol': 'XMR/USD', 'base': 'XMR', 'quote': 'USD', 'baseId': 'tXMR', 'quoteId': 'zUSD' },\n                'XRP/BTC': { 'id': 'tXRPBTC', 'symbol': 'XRP/BTC', 'base': 'XRP', 'quote': 'BTC', 'baseId': 'tXRP', 'quoteId': 'tBTC' },\n                'XRP/USD': { 'id': 'tXRPUSD', 'symbol': 'XRP/USD', 'base': 'XRP', 'quote': 'USD', 'baseId': 'tXRP', 'quoteId': 'zUSD' },\n                'ZEC/BTC': { 'id': 'tZECBTC', 'symbol': 'ZEC/BTC', 'base': 'ZEC', 'quote': 'BTC', 'baseId': 'tZEC', 'quoteId': 'tBTC' },\n                'ZEC/USD': { 'id': 'tZECUSD', 'symbol': 'ZEC/USD', 'base': 'ZEC', 'quote': 'USD', 'baseId': 'tZEC', 'quoteId': 'zUSD' },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.1 / 100,\n                    'taker': 0.2 / 100,\n                },\n                'funding': {\n                    'withdraw': {\n                        'BTC': 0.0005,\n                        'BCH': 0.0005,\n                        'ETH': 0.01,\n                        'EOS': 0.1,\n                        'LTC': 0.001,\n                        'OMG': 0.1,\n                        'IOT': 0.0,\n                        'NEO': 0.0,\n                        'ETC': 0.01,\n                        'XRP': 0.02,\n                        'ETP': 0.01,\n                        'ZEC': 0.001,\n                        'BTG': 0.0,\n                        'DASH': 0.01,\n                        'XMR': 0.04,\n                        'QTM': 0.01,\n                        'EDO': 0.5,\n                        'DAT': 1.0,\n                        'AVT': 0.5,\n                        'SAN': 0.1,\n                        'USDT': 5.0,\n                    },\n                },\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        const currencies = {\n            'DSH': 'DASH', // Bitfinex names Dash as DSH, instead of DASH\n            'QTM': 'QTUM',\n            'BCC': 'CST_BCC',\n            'BCU': 'CST_BCU',\n            'IOT': 'IOTA',\n            'DAT': 'DATA',\n        };\n        return (currency in currencies) ? currencies[currency] : currency;\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostAuthRWallets ();\n        let balanceType = this.safeString (params, 'type', 'exchange');\n        let result = { 'info': response };\n        for (let b = 0; b < response.length; b++) {\n            let balance = response[b];\n            let [ accountType, currency, total, interest, available ] = balance;\n            if (accountType === balanceType) {\n                if (currency[0] === 't')\n                    currency = currency.slice (1);\n                let uppercase = currency.toUpperCase ();\n                uppercase = this.commonCurrencyCode (uppercase);\n                let account = this.account ();\n                account['free'] = available;\n                account['total'] = total;\n                if (account['free'])\n                    account['used'] = account['total'] - account['free'];\n                result[uppercase] = account;\n            }\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetBookSymbolPrecision (this.extend ({\n            'symbol': this.marketId (symbol),\n            'precision': 'R0',\n        }, params));\n        let timestamp = this.milliseconds ();\n        let result = {\n            'bids': [],\n            'asks': [],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n        };\n        for (let i = 0; i < orderbook.length; i++) {\n            let order = orderbook[i];\n            let price = order[1];\n            let amount = order[2];\n            let side = (amount > 0) ? 'bids' : 'asks';\n            amount = Math.abs (amount);\n            result[side].push ([ price, amount ]);\n        }\n        result['bids'] = this.sortBy (result['bids'], 0, true);\n        result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let length = ticker.length;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': ticker[length - 2],\n            'low': ticker[length - 1],\n            'bid': ticker[length - 10],\n            'ask': ticker[length - 8],\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': ticker[length - 4],\n            'change': ticker[length - 6],\n            'percentage': ticker[length - 5],\n            'average': undefined,\n            'baseVolume': ticker[length - 3],\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        let tickers = await this.publicGetTickers (this.extend ({\n            'symbols': this.ids.join (','),\n        }, params));\n        let result = {};\n        for (let i = 0; i < tickers.length; i++) {\n            let ticker = tickers[i];\n            let id = ticker[0];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let market = this.markets[symbol];\n        let ticker = await this.publicGetTickerSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let [ id, timestamp, amount, price ] = trade;\n        let side = (amount < 0) ? 'sell' : 'buy';\n        if (amount < 0) {\n            amount = -amount;\n        }\n        return {\n            'id': id.toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': side,\n            'price': price,\n            'amount': amount,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n        };\n        if (since) {\n            request['start'] = since;\n        }\n        if (limit) {\n            request['limit'] = limit;\n        }\n        let response = await this.publicGetTradesSymbolHist (this.extend (request, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n            'timeframe': this.timeframes[timeframe],\n            'sort': 1,\n        };\n        if (limit)\n            request['limit'] = limit;\n        if (since)\n            request['start'] = since;\n        request = this.extend (request, params);\n        let response = await this.publicGetCandlesTradeTimeframeSymbolHist (request);\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        throw new NotSupported (this.id + ' createOrder not implemented yet');\n    }\n\n    cancelOrder (id, symbol = undefined, params = {}) {\n        throw new NotSupported (this.id + ' cancelOrder not implemented yet');\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchOrder not implemented yet');\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        throw new NotSupported (this.id + ' withdraw not implemented yet');\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let request = this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        let url = this.urls['api'] + '/' + request;\n        if (api === 'public') {\n            if (Object.keys (query).length) {\n                url += '?' + this.urlencode (query);\n            }\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            body = this.json (query);\n            let auth = '/api' + '/' + request + nonce + body;\n            let signature = this.hmac (this.encode (auth), this.encode (this.secret), 'sha384');\n            headers = {\n                'bfx-nonce': nonce,\n                'bfx-apikey': this.apiKey,\n                'bfx-signature': signature,\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (response) {\n            if ('message' in response) {\n                if (response['message'].indexOf ('not enough exchange balance') >= 0)\n                    throw new InsufficientFunds (this.id + ' ' + this.json (response));\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n            }\n            return response;\n        } else if (response === '') {\n            throw new ExchangeError (this.id + ' returned empty response');\n        }\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitflyer extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitflyer',\n            'name': 'bitFlyer',\n            'countries': 'JP',\n            'version': 'v1',\n            'rateLimit': 500,\n            'hasCORS': false,\n            'hasWithdraw': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28051642-56154182-660e-11e7-9b0d-6042d1e6edd8.jpg',\n                'api': 'https://api.bitflyer.jp',\n                'www': 'https://bitflyer.jp',\n                'doc': 'https://bitflyer.jp/API',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'getmarkets',    // or 'markets'\n                        'getboard',      // or 'board'\n                        'getticker',     // or 'ticker'\n                        'getexecutions', // or 'executions'\n                        'gethealth',\n                        'getchats',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'getpermissions',\n                        'getbalance',\n                        'getcollateral',\n                        'getcollateralaccounts',\n                        'getaddresses',\n                        'getcoinins',\n                        'getcoinouts',\n                        'getbankaccounts',\n                        'getdeposits',\n                        'getwithdrawals',\n                        'getchildorders',\n                        'getparentorders',\n                        'getparentorder',\n                        'getexecutions',\n                        'getpositions',\n                        'gettradingcommission',\n                    ],\n                    'post': [\n                        'sendcoin',\n                        'withdraw',\n                        'sendchildorder',\n                        'cancelchildorder',\n                        'sendparentorder',\n                        'cancelparentorder',\n                        'cancelallchildorders',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.25 / 100,\n                    'taker': 0.25 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetMarkets ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['product_code'];\n            let currencies = id.split ('_');\n            let base = undefined;\n            let quote = undefined;\n            let symbol = id;\n            let numCurrencies = currencies.length;\n            if (numCurrencies === 1) {\n                base = symbol.slice (0, 3);\n                quote = symbol.slice (3, 6);\n            } else if (numCurrencies === 2) {\n                base = currencies[0];\n                quote = currencies[1];\n                symbol = base + '/' + quote;\n            } else {\n                base = currencies[1];\n                quote = currencies[2];\n            }\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetBalance ();\n        let balances = {};\n        for (let b = 0; b < response.length; b++) {\n            let account = response[b];\n            let currency = account['currency_code'];\n            balances[currency] = account;\n        }\n        let result = { 'info': response };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            if (currency in balances) {\n                account['total'] = balances[currency]['amount'];\n                account['free'] = balances[currency]['available'];\n                account['used'] = account['total'] - account['free'];\n            }\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetBoard (this.extend ({\n            'product_code': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'size');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let ticker = await this.publicGetTicker (this.extend ({\n            'product_code': this.marketId (symbol),\n        }, params));\n        let timestamp = this.parse8601 (ticker['timestamp']);\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['best_bid']),\n            'ask': parseFloat (ticker['best_ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['ltp']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume_by_product']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let side = undefined;\n        let order = undefined;\n        if ('side' in trade)\n            if (trade['side']) {\n                side = trade['side'].toLowerCase ();\n                let id = side + '_child_order_acceptance_id';\n                if (id in trade)\n                    order = trade[id];\n            }\n        let timestamp = this.parse8601 (trade['exec_date']);\n        return {\n            'id': trade['id'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': order,\n            'type': undefined,\n            'side': side,\n            'price': trade['price'],\n            'amount': trade['size'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetExecutions (this.extend ({\n            'product_code': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'product_code': this.marketId (symbol),\n            'child_order_type': type.toUpperCase (),\n            'side': side.toUpperCase (),\n            'price': price,\n            'size': amount,\n        };\n        let result = await this.privatePostSendchildorder (this.extend (order, params));\n        return {\n            'info': result,\n            'id': result['child_order_acceptance_id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelchildorder (this.extend ({\n            'parent_order_id': id,\n        }, params));\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostWithdraw (this.extend ({\n            'currency_code': currency,\n            'amount': amount,\n            // 'bank_account_id': 1234,\n        }, params));\n        return {\n            'info': response,\n            'id': response['message_id'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let request = '/' + this.version + '/';\n        if (api === 'private')\n            request += 'me/';\n        request += path;\n        if (method === 'GET') {\n            if (Object.keys (params).length)\n                request += '?' + this.urlencode (params);\n        }\n        let url = this.urls['api'] + request;\n        if (api === 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            body = this.json (params);\n            let auth = [ nonce, method, request, body ].join ('');\n            headers = {\n                'ACCESS-KEY': this.apiKey,\n                'ACCESS-TIMESTAMP': nonce,\n                'ACCESS-SIGN': this.hmac (this.encode (auth), this.encode (this.secret)),\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bithumb extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bithumb',\n            'name': 'Bithumb',\n            'countries': 'KR', // South Korea\n            'rateLimit': 500,\n            'hasCORS': true,\n            // obsolete metainfo interface\n            'hasFetchTickers': true,\n            'hasWithdraw': true,\n            // new metainfo interface\n            'has': {\n                'fetchTickers': true,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30597177-ea800172-9d5e-11e7-804c-b9d4fa9b56b0.jpg',\n                'api': {\n                    'public': 'https://api.bithumb.com/public',\n                    'private': 'https://api.bithumb.com',\n                },\n                'www': 'https://www.bithumb.com',\n                'doc': 'https://www.bithumb.com/u1/US127',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'ticker/{currency}',\n                        'ticker/all',\n                        'orderbook/{currency}',\n                        'orderbook/all',\n                        'recent_transactions/{currency}',\n                        'recent_transactions/all',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'info/account',\n                        'info/balance',\n                        'info/wallet_address',\n                        'info/ticker',\n                        'info/orders',\n                        'info/user_transactions',\n                        'trade/place',\n                        'info/order_detail',\n                        'trade/cancel',\n                        'trade/btc_withdrawal',\n                        'trade/krw_deposit',\n                        'trade/krw_withdrawal',\n                        'trade/market_buy',\n                        'trade/market_sell',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.15 / 100,\n                    'taker': 0.15 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetTickerAll ();\n        let currencies = Object.keys (markets['data']);\n        let result = [];\n        for (let i = 0; i < currencies.length; i++) {\n            let id = currencies[i];\n            if (id !== 'date') {\n                let market = markets['data'][id];\n                let base = id;\n                let quote = 'KRW';\n                let symbol = id + '/' + quote;\n                result.push (this.extend (this.fees['trading'], {\n                    'id': id,\n                    'symbol': symbol,\n                    'base': base,\n                    'quote': quote,\n                    'info': market,\n                    'lot': undefined,\n                    'active': true,\n                    'precision': {\n                        'amount': undefined,\n                        'price': undefined,\n                    },\n                    'limits': {\n                        'amount': {\n                            'min': undefined,\n                            'max': undefined,\n                        },\n                        'price': {\n                            'min': undefined,\n                            'max': undefined,\n                        },\n                        'cost': {\n                            'min': undefined,\n                            'max': undefined,\n                        },\n                    },\n                }));\n            }\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostInfoBalance (this.extend ({\n            'currency': 'ALL',\n        }, params));\n        let result = { 'info': response };\n        let balances = response['data'];\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            let lowercase = currency.toLowerCase ();\n            account['total'] = this.safeFloat (balances, 'total_' + lowercase);\n            account['used'] = this.safeFloat (balances, 'in_use_' + lowercase);\n            account['free'] = this.safeFloat (balances, 'available_' + lowercase);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetOrderbookCurrency (this.extend ({\n            'count': 50, // max = 50\n            'currency': market['base'],\n        }, params));\n        let orderbook = response['data'];\n        let timestamp = parseInt (orderbook['timestamp']);\n        return this.parseOrderBook (orderbook, timestamp, 'bids', 'asks', 'price', 'quantity');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = parseInt (ticker['date']);\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'max_price'),\n            'low': this.safeFloat (ticker, 'min_price'),\n            'bid': this.safeFloat (ticker, 'buy_price'),\n            'ask': this.safeFloat (ticker, 'sell_price'),\n            'vwap': undefined,\n            'open': this.safeFloat (ticker, 'opening_price'),\n            'close': this.safeFloat (ticker, 'closing_price'),\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last_trade'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': this.safeFloat (ticker, 'average_price'),\n            'baseVolume': this.safeFloat (ticker, 'volume_1day'),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetTickerAll (params);\n        let result = {};\n        let timestamp = response['data']['date'];\n        let tickers = this.omit (response['data'], 'date');\n        let ids = Object.keys (tickers);\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let symbol = id;\n            let market = undefined;\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            }\n            let ticker = tickers[id];\n            ticker['date'] = timestamp;\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTickerCurrency (this.extend ({\n            'currency': market['base'],\n        }, params));\n        return this.parseTicker (response['data'], market);\n    }\n\n    parseTrade (trade, market) {\n        // a workaround for their bug in date format, hours are not 0-padded\n        let [ transaction_date, transaction_time ] = trade['transaction_date'].split (' ');\n        let transaction_time_short = transaction_time.length < 8;\n        if (transaction_time_short)\n            transaction_time = '0' + transaction_time;\n        let timestamp = this.parse8601 (transaction_date + ' ' + transaction_time);\n        let side = (trade['type'] === 'ask') ? 'sell' : 'buy';\n        return {\n            'id': undefined,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': undefined,\n            'type': undefined,\n            'side': side,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['units_traded']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetRecentTransactionsCurrency (this.extend ({\n            'currency': market['base'],\n            'count': 100, // max = 100\n        }, params));\n        return this.parseTrades (response['data'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = undefined;\n        let method = 'privatePostTrade';\n        if (type === 'limit') {\n            request = {\n                'order_currency': market['id'],\n                'Payment_currency': market['quote'],\n                'units': amount,\n                'price': price,\n                'type': (side === 'buy') ? 'bid' : 'ask',\n            };\n            method += 'Place';\n        } else if (type === 'market') {\n            request = {\n                'currency': market['id'],\n                'units': amount,\n            };\n            method += 'Market' + this.capitalize (side);\n        }\n        let response = await this[method] (this.extend (request, params));\n        let id = undefined;\n        if ('order_id' in response) {\n            if (response['order_id'])\n                id = response['order_id'].toString ();\n        }\n        return {\n            'info': response,\n            'id': id,\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        let side = ('side' in params);\n        if (!side)\n            throw new ExchangeError (this.id + ' cancelOrder requires a side parameter (sell or buy) and a currency parameter');\n        side = (side === 'buy') ? 'purchase' : 'sales';\n        let currency = ('currency' in params);\n        if (!currency)\n            throw new ExchangeError (this.id + ' cancelOrder requires a currency parameter');\n        return await this.privatePostTradeCancel ({\n            'order_id': id,\n            'type': params['side'],\n            'currency': params['currency'],\n        });\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        let request = {\n            'units': amount,\n            'address': address,\n            'currency': currency,\n        };\n        if (currency === 'XRP' || currency === 'XMR') {\n            let destination = ('destination' in params);\n            if (!destination)\n                throw new ExchangeError (this.id + ' ' + currency + ' withdraw requires an extra destination param');\n        }\n        let response = await this.privatePostTradeBtcWithdrawal (this.extend (request, params));\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let endpoint = '/' + this.implodeParams (path, params);\n        let url = this.urls['api'][api] + endpoint;\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            body = this.urlencode (this.extend ({\n                'endpoint': endpoint,\n            }, query));\n            let nonce = this.nonce ().toString ();\n            let auth = endpoint + '\\0' + body + '\\0' + nonce;\n            let signature = this.hmac (this.encode (auth), this.encode (this.secret), 'sha512');\n            let signature64 = this.decode (this.stringToBase64 (this.encode (signature)));\n            headers = {\n                'Accept': 'application/json',\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Api-Key': this.apiKey,\n                'Api-Sign': signature64.toString (),\n                'Api-Nonce': nonce,\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('status' in response) {\n            if (response['status'] === '0000')\n                return response;\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { NotSupported } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitlish extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitlish',\n            'name': 'Bitlish',\n            'countries': [ 'GB', 'EU', 'RU' ],\n            'rateLimit': 1500,\n            'version': 'v1',\n            'hasCORS': false,\n            'hasFetchTickers': true,\n            'hasFetchOHLCV': true,\n            'hasWithdraw': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766275-dcfc6c30-5ed3-11e7-839d-00a846385d0b.jpg',\n                'api': 'https://bitlish.com/api',\n                'www': 'https://bitlish.com',\n                'doc': 'https://bitlish.com/api',\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': false,\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'taker': 0.3 / 100, // anonymous 0.3%, verified 0.2%\n                    'maker': 0,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.001,\n                        'LTC': 0.001,\n                        'DOGE': 0.001,\n                        'ETH': 0.001,\n                        'XMR': 0,\n                        'ZEC': 0.001,\n                        'DASH': 0.0001,\n                        'EUR': 50,\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                        'LTC': 0,\n                        'DOGE': 0,\n                        'ETH': 0,\n                        'XMR': 0,\n                        'ZEC': 0,\n                        'DASH': 0,\n                        'EUR': 0,\n                    },\n                },\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'instruments',\n                        'ohlcv',\n                        'pairs',\n                        'tickers',\n                        'trades_depth',\n                        'trades_history',\n                    ],\n                    'post': [\n                        'instruments',\n                        'ohlcv',\n                        'pairs',\n                        'tickers',\n                        'trades_depth',\n                        'trades_history',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'accounts_operations',\n                        'balance',\n                        'cancel_trade',\n                        'cancel_trades_by_ids',\n                        'cancel_all_trades',\n                        'create_bcode',\n                        'create_template_wallet',\n                        'create_trade',\n                        'deposit',\n                        'list_accounts_operations_from_ts',\n                        'list_active_trades',\n                        'list_bcodes',\n                        'list_my_matches_from_ts',\n                        'list_my_trades',\n                        'list_my_trads_from_ts',\n                        'list_payment_methods',\n                        'list_payments',\n                        'redeem_code',\n                        'resign',\n                        'signin',\n                        'signout',\n                        'trade_details',\n                        'trade_options',\n                        'withdraw',\n                        'withdraw_by_id',\n                    ],\n                },\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        if (!this.substituteCommonCurrencyCodes)\n            return currency;\n        if (currency === 'XBT')\n            return 'BTC';\n        if (currency === 'BCC')\n            return 'BCH';\n        if (currency === 'DRK')\n            return 'DASH';\n        if (currency === 'DSH')\n            currency = 'DASH';\n        if (currency === 'XDG')\n            currency = 'DOGE';\n        return currency;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetPairs ();\n        let result = [];\n        let keys = Object.keys (markets);\n        for (let p = 0; p < keys.length; p++) {\n            let market = markets[keys[p]];\n            let id = market['id'];\n            let symbol = market['name'];\n            let [ base, quote ] = symbol.split ('/');\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    parseTicker (ticker, market) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'high': this.safeFloat (ticker, 'max'),\n            'low': this.safeFloat (ticker, 'min'),\n            'bid': undefined,\n            'ask': undefined,\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': this.safeFloat (ticker, 'first'),\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': this.safeFloat (ticker, 'prc'),\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'sum'),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTickers (params);\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let tickers = await this.publicGetTickers (params);\n        let ticker = tickers[market['id']];\n        return this.parseTicker (ticker, market);\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        // let market = this.market (symbol);\n        let now = this.seconds ();\n        let start = now - 86400 * 30; // last 30 days\n        let interval = [ start.toString (), undefined ];\n        return await this.publicPostOhlcv (this.extend ({\n            'time_range': interval,\n        }, params));\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetTradesDepth (this.extend ({\n            'pair_id': this.marketId (symbol),\n        }, params));\n        let timestamp = undefined;\n        let last = this.safeInteger (orderbook, 'last');\n        if (last)\n            timestamp = parseInt (last / 1000);\n        return this.parseOrderBook (orderbook, timestamp, 'bid', 'ask', 'price', 'volume');\n    }\n\n    parseTrade (trade, market = undefined) {\n        let side = (trade['dir'] === 'bid') ? 'buy' : 'sell';\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = parseInt (trade['created'] / 1000);\n        return {\n            'id': undefined,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'order': undefined,\n            'type': undefined,\n            'side': side,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTradesHistory (this.extend ({\n            'pair_id': market['id'],\n        }, params));\n        return this.parseTrades (response['list'], market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostBalance ();\n        let result = { 'info': response };\n        let currencies = Object.keys (response);\n        let balance = {};\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let account = response[currency];\n            currency = currency.toUpperCase ();\n            // issue #4 bitlish names Dash as DSH, instead of DASH\n            if (currency === 'DSH')\n                currency = 'DASH';\n            if (currency === 'XDG')\n                currency = 'DOGE';\n            balance[currency] = account;\n        }\n        currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            if (currency in balance) {\n                account['free'] = parseFloat (balance[currency]['funds']);\n                account['used'] = parseFloat (balance[currency]['holded']);\n                account['total'] = this.sum (account['free'], account['used']);\n            }\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    signIn () {\n        return this.privatePostSignin ({\n            'login': this.login,\n            'passwd': this.password,\n        });\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'pair_id': this.marketId (symbol),\n            'dir': (side === 'buy') ? 'bid' : 'ask',\n            'amount': amount,\n        };\n        if (type === 'limit')\n            order['price'] = price;\n        let result = await this.privatePostCreateTrade (this.extend (order, params));\n        return {\n            'info': result,\n            'id': result['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelTrade ({ 'id': id });\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        if (currency !== 'BTC') {\n            // they did not document other types...\n            throw new NotSupported (this.id + ' currently supports BTC withdrawals only, until they document other currencies...');\n        }\n        let response = await this.privatePostWithdraw (this.extend ({\n            'currency': currency.toLowerCase (),\n            'amount': parseFloat (amount),\n            'account': address,\n            'payment_method': 'bitcoin', // they did not document other types...\n        }, params));\n        return {\n            'info': response,\n            'id': response['message_id'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + path;\n        if (api === 'public') {\n            if (method === 'GET') {\n                if (Object.keys (params).length)\n                    url += '?' + this.urlencode (params);\n            } else {\n                body = this.json (params);\n                headers = { 'Content-Type': 'application/json' };\n            }\n        } else {\n            this.checkRequiredCredentials ();\n            body = this.json (this.extend ({ 'token': this.apiKey }, params));\n            headers = { 'Content-Type': 'application/json' };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitmarket extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitmarket',\n            'name': 'BitMarket',\n            'countries': [ 'PL', 'EU' ],\n            'rateLimit': 1500,\n            'hasCORS': false,\n            'hasFetchOHLCV': true,\n            'hasWithdraw': true,\n            'timeframes': {\n                '90m': '90m',\n                '6h': '6h',\n                '1d': '1d',\n                '1w': '7d',\n                '1M': '1m',\n                '3M': '3m',\n                '6M': '6m',\n                '1y': '1y',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27767256-a8555200-5ef9-11e7-96fd-469a65e2b0bd.jpg',\n                'api': {\n                    'public': 'https://www.bitmarket.net',\n                    'private': 'https://www.bitmarket.pl/api2/', // last slash is critical\n                },\n                'www': [\n                    'https://www.bitmarket.pl',\n                    'https://www.bitmarket.net',\n                ],\n                'doc': [\n                    'https://www.bitmarket.net/docs.php?file=api_public.html',\n                    'https://www.bitmarket.net/docs.php?file=api_private.html',\n                    'https://github.com/bitmarket-net/api',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'json/{market}/ticker',\n                        'json/{market}/orderbook',\n                        'json/{market}/trades',\n                        'json/ctransfer',\n                        'graphs/{market}/90m',\n                        'graphs/{market}/6h',\n                        'graphs/{market}/1d',\n                        'graphs/{market}/7d',\n                        'graphs/{market}/1m',\n                        'graphs/{market}/3m',\n                        'graphs/{market}/6m',\n                        'graphs/{market}/1y',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'info',\n                        'trade',\n                        'cancel',\n                        'orders',\n                        'trades',\n                        'history',\n                        'withdrawals',\n                        'tradingdesk',\n                        'tradingdeskStatus',\n                        'tradingdeskConfirm',\n                        'cryptotradingdesk',\n                        'cryptotradingdeskStatus',\n                        'cryptotradingdeskConfirm',\n                        'withdraw',\n                        'withdrawFiat',\n                        'withdrawPLNPP',\n                        'withdrawFiatFast',\n                        'deposit',\n                        'transfer',\n                        'transfers',\n                        'marginList',\n                        'marginOpen',\n                        'marginClose',\n                        'marginCancel',\n                        'marginModify',\n                        'marginBalanceAdd',\n                        'marginBalanceRemove',\n                        'swapList',\n                        'swapOpen',\n                        'swapClose',\n                    ],\n                },\n            },\n            'markets': {\n                'BCH/PLN': { 'id': 'BCCPLN', 'symbol': 'BCH/PLN', 'base': 'BCH', 'quote': 'PLN' },\n                'BTG/PLN': { 'id': 'BTGPLN', 'symbol': 'BTG/PLN', 'base': 'BTG', 'quote': 'PLN' },\n                'BTC/PLN': { 'id': 'BTCPLN', 'symbol': 'BTC/PLN', 'base': 'BTC', 'quote': 'PLN' },\n                'BTC/EUR': { 'id': 'BTCEUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR' },\n                'LTC/PLN': { 'id': 'LTCPLN', 'symbol': 'LTC/PLN', 'base': 'LTC', 'quote': 'PLN' },\n                'LTC/BTC': { 'id': 'LTCBTC', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC' },\n                'LiteMineX/BTC': { 'id': 'LiteMineXBTC', 'symbol': 'LiteMineX/BTC', 'base': 'LiteMineX', 'quote': 'BTC' },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': true,\n                    'percentage': true,\n                    'taker': 0.45 / 100,\n                    'maker': 0.15 / 100,\n                    'tiers': {\n                        'taker': [\n                            [0, 0.45 / 100],\n                            [99.99, 0.44 / 100],\n                            [299.99, 0.43 / 100],\n                            [499.99, 0.42 / 100],\n                            [999.99, 0.41 / 100],\n                            [1999.99, 0.40 / 100],\n                            [2999.99, 0.39 / 100],\n                            [4999.99, 0.38 / 100],\n                            [9999.99, 0.37 / 100],\n                            [19999.99, 0.36 / 100],\n                            [29999.99, 0.35 / 100],\n                            [49999.99, 0.34 / 100],\n                            [99999.99, 0.33 / 100],\n                            [199999.99, 0.32 / 100],\n                            [299999.99, 0.31 / 100],\n                            [499999.99, 0.0 / 100],\n                        ],\n                        'maker': [\n                            [0, 0.15 / 100],\n                            [99.99, 0.14 / 100],\n                            [299.99, 0.13 / 100],\n                            [499.99, 0.12 / 100],\n                            [999.99, 0.11 / 100],\n                            [1999.99, 0.10 / 100],\n                            [2999.99, 0.9 / 100],\n                            [4999.99, 0.8 / 100],\n                            [9999.99, 0.7 / 100],\n                            [19999.99, 0.6 / 100],\n                            [29999.99, 0.5 / 100],\n                            [49999.99, 0.4 / 100],\n                            [99999.99, 0.3 / 100],\n                            [199999.99, 0.2 / 100],\n                            [299999.99, 0.1 / 100],\n                            [499999.99, 0.0 / 100],\n                        ],\n                    },\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.0008,\n                        'LTC': 0.005,\n                        'BCH': 0.0008,\n                        'BTG': 0.0008,\n                        'DOGE': 1,\n                        'EUR': 2,\n                        'PLN': 2,\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                        'LTC': 0,\n                        'BCH': 0,\n                        'BTG': 0,\n                        'DOGE': 25,\n                        'EUR': 2, // SEPA. Transfer INT (SHA): 5 EUR\n                        'PLN': 0,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostInfo ();\n        let data = response['data'];\n        let balance = data['balances'];\n        let result = { 'info': data };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            if (currency in balance['available'])\n                account['free'] = balance['available'][currency];\n            if (currency in balance['blocked'])\n                account['used'] = balance['blocked'][currency];\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetJsonMarketOrderbook (this.extend ({\n            'market': this.marketId (symbol),\n        }, params));\n        let timestamp = this.milliseconds ();\n        return {\n            'bids': orderbook['bids'],\n            'asks': orderbook['asks'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetJsonMarketTicker (this.extend ({\n            'market': this.marketId (symbol),\n        }, params));\n        let timestamp = this.milliseconds ();\n        let vwap = parseFloat (ticker['vwap']);\n        let baseVolume = parseFloat (ticker['volume']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': vwap,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let side = (trade['type'] === 'bid') ? 'buy' : 'sell';\n        let timestamp = trade['date'] * 1000;\n        return {\n            'id': trade['tid'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': undefined,\n            'type': undefined,\n            'side': side,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetJsonMarketTrades (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '90m', since = undefined, limit = undefined) {\n        return [\n            ohlcv['time'] * 1000,\n            parseFloat (ohlcv['open']),\n            parseFloat (ohlcv['high']),\n            parseFloat (ohlcv['low']),\n            parseFloat (ohlcv['close']),\n            parseFloat (ohlcv['vol']),\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '90m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let method = 'publicGetGraphsMarket' + this.timeframes[timeframe];\n        let market = this.market (symbol);\n        let response = await this[method] (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let response = await this.privatePostTrade (this.extend ({\n            'market': this.marketId (symbol),\n            'type': side,\n            'amount': amount,\n            'rate': price,\n        }, params));\n        let result = {\n            'info': response,\n        };\n        if ('id' in response['order'])\n            result['id'] = response['id'];\n        return result;\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancel ({ 'id': id });\n    }\n\n    isFiat (currency) {\n        if (currency === 'EUR')\n            return true;\n        if (currency === 'PLN')\n            return true;\n        return false;\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let method = undefined;\n        let request = {\n            'currency': currency,\n            'quantity': amount,\n        };\n        if (this.isFiat (currency)) {\n            method = 'privatePostWithdrawFiat';\n            if ('account' in params) {\n                request['account'] = params['account']; // bank account code for withdrawal\n            } else {\n                throw new ExchangeError (this.id + ' requires account parameter to withdraw fiat currency');\n            }\n            if ('account2' in params) {\n                request['account2'] = params['account2']; // bank SWIFT code (EUR only)\n            } else {\n                if (currency === 'EUR')\n                    throw new ExchangeError (this.id + ' requires account2 parameter to withdraw EUR');\n            }\n            if ('withdrawal_note' in params) {\n                request['withdrawal_note'] = params['withdrawal_note']; // a 10-character user-specified withdrawal note (PLN only)\n            } else {\n                if (currency === 'PLN')\n                    throw new ExchangeError (this.id + ' requires withdrawal_note parameter to withdraw PLN');\n            }\n        } else {\n            method = 'privatePostWithdraw';\n            request['address'] = address;\n        }\n        let response = await this[method] (this.extend (request, params));\n        return {\n            'info': response,\n            'id': response,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        if (api === 'public') {\n            url += '/' + this.implodeParams (path + '.json', params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let query = this.extend ({\n                'tonce': nonce,\n                'method': path,\n            }, params);\n            body = this.urlencode (query);\n            headers = {\n                'API-Key': this.apiKey,\n                'API-Hash': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, DDoSProtection } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitmex extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitmex',\n            'name': 'BitMEX',\n            'countries': 'SC', // Seychelles\n            'version': 'v1',\n            'userAgent': undefined,\n            'rateLimit': 1500,\n            'hasCORS': false,\n            'hasFetchOHLCV': true,\n            'hasWithdraw': true,\n            'timeframes': {\n                '1m': '1m',\n                '5m': '5m',\n                '1h': '1h',\n                '1d': '1d',\n            },\n            'urls': {\n                'test': 'https://testnet.bitmex.com',\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766319-f653c6e6-5ed4-11e7-933d-f0bc3699ae8f.jpg',\n                'api': 'https://www.bitmex.com',\n                'www': 'https://www.bitmex.com',\n                'doc': [\n                    'https://www.bitmex.com/app/apiOverview',\n                    'https://github.com/BitMEX/api-connectors/tree/master/official-http',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'announcement',\n                        'announcement/urgent',\n                        'funding',\n                        'instrument',\n                        'instrument/active',\n                        'instrument/activeAndIndices',\n                        'instrument/activeIntervals',\n                        'instrument/compositeIndex',\n                        'instrument/indices',\n                        'insurance',\n                        'leaderboard',\n                        'liquidation',\n                        'orderBook',\n                        'orderBook/L2',\n                        'quote',\n                        'quote/bucketed',\n                        'schema',\n                        'schema/websocketHelp',\n                        'settlement',\n                        'stats',\n                        'stats/history',\n                        'trade',\n                        'trade/bucketed',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'apiKey',\n                        'chat',\n                        'chat/channels',\n                        'chat/connected',\n                        'execution',\n                        'execution/tradeHistory',\n                        'notification',\n                        'order',\n                        'position',\n                        'user',\n                        'user/affiliateStatus',\n                        'user/checkReferralCode',\n                        'user/commission',\n                        'user/depositAddress',\n                        'user/margin',\n                        'user/minWithdrawalFee',\n                        'user/wallet',\n                        'user/walletHistory',\n                        'user/walletSummary',\n                    ],\n                    'post': [\n                        'apiKey',\n                        'apiKey/disable',\n                        'apiKey/enable',\n                        'chat',\n                        'order',\n                        'order/bulk',\n                        'order/cancelAllAfter',\n                        'order/closePosition',\n                        'position/isolate',\n                        'position/leverage',\n                        'position/riskLimit',\n                        'position/transferMargin',\n                        'user/cancelWithdrawal',\n                        'user/confirmEmail',\n                        'user/confirmEnableTFA',\n                        'user/confirmWithdrawal',\n                        'user/disableTFA',\n                        'user/logout',\n                        'user/logoutAll',\n                        'user/preferences',\n                        'user/requestEnableTFA',\n                        'user/requestWithdrawal',\n                    ],\n                    'put': [\n                        'order',\n                        'order/bulk',\n                        'user',\n                    ],\n                    'delete': [\n                        'apiKey',\n                        'order',\n                        'order/all',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetInstrumentActiveAndIndices ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let active = (market['state'] !== 'Unlisted');\n            let id = market['symbol'];\n            let base = market['underlying'];\n            let quote = market['quoteCurrency'];\n            let type = undefined;\n            let future = false;\n            let prediction = false;\n            let basequote = base + quote;\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let swap = (id === basequote);\n            let symbol = id;\n            if (swap) {\n                type = 'swap';\n                symbol = base + '/' + quote;\n            } else if (id.indexOf ('B_') >= 0) {\n                prediction = true;\n                type = 'prediction';\n            } else {\n                future = true;\n                type = 'future';\n            }\n            let maker = market['makerFee'];\n            let taker = market['takerFee'];\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': active,\n                'taker': taker,\n                'maker': maker,\n                'type': type,\n                'spot': false,\n                'swap': swap,\n                'future': future,\n                'prediction': prediction,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetUserMargin ({ 'currency': 'all' });\n        let result = { 'info': response };\n        for (let b = 0; b < response.length; b++) {\n            let balance = response[b];\n            let currency = balance['currency'].toUpperCase ();\n            currency = this.commonCurrencyCode (currency);\n            let account = {\n                'free': balance['availableMargin'],\n                'used': 0.0,\n                'total': balance['marginBalance'],\n            };\n            if (currency === 'BTC') {\n                account['free'] = account['free'] * 0.00000001;\n                account['total'] = account['total'] * 0.00000001;\n            }\n            account['used'] = account['total'] - account['free'];\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderBookL2 (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        let timestamp = this.milliseconds ();\n        let result = {\n            'bids': [],\n            'asks': [],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n        };\n        for (let o = 0; o < orderbook.length; o++) {\n            let order = orderbook[o];\n            let side = (order['side'] === 'Sell') ? 'asks' : 'bids';\n            let amount = order['size'];\n            let price = order['price'];\n            result[side].push ([ price, amount ]);\n        }\n        result['bids'] = this.sortBy (result['bids'], 0, true);\n        result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        if (!market['active'])\n            throw new ExchangeError (this.id + ': symbol ' + symbol + ' is delisted');\n        let request = this.extend ({\n            'symbol': market['id'],\n            'binSize': '1d',\n            'partial': true,\n            'count': 1,\n            'reverse': true,\n        }, params);\n        let quotes = await this.publicGetQuoteBucketed (request);\n        let quotesLength = quotes.length;\n        let quote = quotes[quotesLength - 1];\n        let tickers = await this.publicGetTradeBucketed (request);\n        let ticker = tickers[0];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (quote['bidPrice']),\n            'ask': parseFloat (quote['askPrice']),\n            'vwap': parseFloat (ticker['vwap']),\n            'open': undefined,\n            'close': parseFloat (ticker['close']),\n            'first': undefined,\n            'last': undefined,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['homeNotional']),\n            'quoteVolume': parseFloat (ticker['foreignNotional']),\n            'info': ticker,\n        };\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        let timestamp = this.parse8601 (ohlcv['timestamp']);\n        return [\n            timestamp,\n            ohlcv['open'],\n            ohlcv['high'],\n            ohlcv['low'],\n            ohlcv['close'],\n            ohlcv['volume'],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        // send JSON key/value pairs, such as {\"key\": \"value\"}\n        // filter by individual fields and do advanced queries on timestamps\n        // let filter = { 'key': 'value' };\n        // send a bare series (e.g. XBU) to nearest expiring contract in that series\n        // you can also send a timeframe, e.g. XBU:monthly\n        // timeframes: daily, weekly, monthly, quarterly, and biquarterly\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n            'binSize': this.timeframes[timeframe],\n            'partial': true,     // true == include yet-incomplete current bins\n            // 'filter': filter, // filter by individual fields and do advanced queries\n            // 'columns': [],    // will return all columns if omitted\n            // 'start': 0,       // starting point for results (wtf?)\n            // 'reverse': false, // true == newest first\n            // 'endTime': '',    // ending date filter for results\n        };\n        if (since) {\n            let ymdhms = this.YmdHMS (since);\n            let ymdhm = ymdhms.slice (0, 16);\n            request['startTime'] = ymdhm; // starting date filter for results\n        }\n        if (limit)\n            request['count'] = limit; // default 100\n        let response = await this.publicGetTradeBucketed (this.extend (request, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = this.parse8601 (trade['timestamp']);\n        let symbol = undefined;\n        if (!market) {\n            if ('symbol' in trade)\n                market = this.markets_by_id[trade['symbol']];\n        }\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'id': trade['trdMatchID'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'order': undefined,\n            'type': undefined,\n            'side': trade['side'].toLowerCase (),\n            'price': trade['price'],\n            'amount': trade['size'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTrade (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'symbol': this.marketId (symbol),\n            'side': this.capitalize (side),\n            'orderQty': amount,\n            'ordType': this.capitalize (type),\n        };\n        if (type === 'limit')\n            order['price'] = price;\n        let response = await this.privatePostOrder (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['orderID'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privateDeleteOrder ({ 'orderID': id });\n    }\n\n    isFiat (currency) {\n        if (currency === 'EUR')\n            return true;\n        if (currency === 'PLN')\n            return true;\n        return false;\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        if (currency !== 'BTC')\n            throw new ExchangeError (this.id + ' supoprts BTC withdrawals only, other currencies coming soon...');\n        let request = {\n            'currency': 'XBt', // temporarily\n            'amount': amount,\n            'address': address,\n            // 'otpToken': '123456', // requires if two-factor auth (OTP) is enabled\n            // 'fee': 0.001, // bitcoin network fee\n        };\n        let response = await this.privatePostUserRequestWithdrawal (this.extend (request, params));\n        return {\n            'info': response,\n            'id': response['transactID'],\n        };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code === 429)\n            throw new DDoSProtection (this.id + ' ' + body);\n        if (code >= 400) {\n            if (body) {\n                if (body[0] === '{') {\n                    let response = JSON.parse (body);\n                    if ('error' in response) {\n                        if ('message' in response['error']) {\n                            // stub code, need proper handling\n                            throw new ExchangeError (this.id + ' ' + this.json (response));\n                        }\n                    }\n                }\n            }\n        }\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let query = '/api' + '/' + this.version + '/' + path;\n        if (method !== 'PUT')\n            if (Object.keys (params).length)\n                query += '?' + this.urlencode (params);\n        let url = this.urls['api'] + query;\n        if (api === 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let auth = method + query + nonce;\n            if (method === 'POST' || method === 'PUT') {\n                if (Object.keys (params).length) {\n                    body = this.json (params);\n                    auth += body;\n                }\n            }\n            headers = {\n                'Content-Type': 'application/json',\n                'api-nonce': nonce,\n                'api-key': this.apiKey,\n                'api-signature': this.hmac (this.encode (auth), this.encode (this.secret)),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitso extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitso',\n            'name': 'Bitso',\n            'countries': 'MX', // Mexico\n            'rateLimit': 2000, // 30 requests per minute\n            'version': 'v3',\n            'hasCORS': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766335-715ce7aa-5ed5-11e7-88a8-173a27bb30fe.jpg',\n                'api': 'https://api.bitso.com',\n                'www': 'https://bitso.com',\n                'doc': 'https://bitso.com/api_info',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'available_books',\n                        'ticker',\n                        'order_book',\n                        'trades',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'account_status',\n                        'balance',\n                        'fees',\n                        'fundings',\n                        'fundings/{fid}',\n                        'funding_destination',\n                        'kyc_documents',\n                        'ledger',\n                        'ledger/trades',\n                        'ledger/fees',\n                        'ledger/fundings',\n                        'ledger/withdrawals',\n                        'mx_bank_codes',\n                        'open_orders',\n                        'order_trades/{oid}',\n                        'orders/{oid}',\n                        'user_trades',\n                        'user_trades/{tid}',\n                        'withdrawals/',\n                        'withdrawals/{wid}',\n                    ],\n                    'post': [\n                        'bitcoin_withdrawal',\n                        'debit_card_withdrawal',\n                        'ether_withdrawal',\n                        'orders',\n                        'phone_number',\n                        'phone_verification',\n                        'phone_withdrawal',\n                        'spei_withdrawal',\n                    ],\n                    'delete': [\n                        'orders/{oid}',\n                        'orders/all',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetAvailableBooks ();\n        let result = [];\n        for (let i = 0; i < markets['payload'].length; i++) {\n            let market = markets['payload'][i];\n            let id = market['book'];\n            let symbol = id.toUpperCase ().replace ('_', '/');\n            let [ base, quote ] = symbol.split ('/');\n            let limits = {\n                'amount': {\n                    'min': parseFloat (market['minimum_amount']),\n                    'max': parseFloat (market['maximum_amount']),\n                },\n                'price': {\n                    'min': parseFloat (market['minimum_price']),\n                    'max': parseFloat (market['maximum_price']),\n                },\n                'cost': {\n                    'min': parseFloat (market['minimum_value']),\n                    'max': parseFloat (market['maximum_value']),\n                },\n            };\n            let precision = {\n                'amount': this.precisionFromString (market['minimum_amount']),\n                'price': this.precisionFromString (market['minimum_price']),\n            };\n            let lot = limits['amount']['min'];\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n                'lot': lot,\n                'limits': limits,\n                'precision': precision,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetBalance ();\n        let balances = response['payload']['balances'];\n        let result = { 'info': response };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'].toUpperCase ();\n            let account = {\n                'free': parseFloat (balance['available']),\n                'used': parseFloat (balance['locked']),\n                'total': parseFloat (balance['total']),\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetOrderBook (this.extend ({\n            'book': this.marketId (symbol),\n        }, params));\n        let orderbook = response['payload'];\n        let timestamp = this.parse8601 (orderbook['updated_at']);\n        return this.parseOrderBook (orderbook, timestamp, 'bids', 'asks', 'price', 'amount');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetTicker (this.extend ({\n            'book': this.marketId (symbol),\n        }, params));\n        let ticker = response['payload'];\n        let timestamp = this.parse8601 (ticker['created_at']);\n        let vwap = parseFloat (ticker['vwap']);\n        let baseVolume = parseFloat (ticker['volume']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': vwap,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = this.parse8601 (trade['created_at']);\n        let symbol = undefined;\n        if (!market) {\n            if ('book' in trade)\n                market = this.markets_by_id[trade['book']];\n        }\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'id': trade['tid'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'order': undefined,\n            'type': undefined,\n            'side': trade['maker_side'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTrades (this.extend ({\n            'book': market['id'],\n        }, params));\n        return this.parseTrades (response['payload'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'book': this.marketId (symbol),\n            'side': side,\n            'type': type,\n            'major': this.amountToPrecision (symbol, amount),\n        };\n        if (type === 'limit')\n            order['price'] = this.priceToPrecision (symbol, price);\n        let response = await this.privatePostOrders (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['payload']['oid'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privateDeleteOrders ({ 'oid': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let query = '/' + this.version + '/' + this.implodeParams (path, params);\n        let url = this.urls['api'] + query;\n        if (api === 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let request = [ nonce, method, query ].join ('');\n            if (Object.keys (params).length) {\n                body = this.json (params);\n                request += body;\n            }\n            let signature = this.hmac (this.encode (request), this.encode (this.secret));\n            let auth = this.apiKey + ':' + nonce + ':' + signature;\n            headers = {\n                'Authorization': 'Bitso ' + auth,\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('success' in response)\n            if (response['success'])\n                return response;\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitstamp extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitstamp',\n            'name': 'Bitstamp',\n            'countries': 'GB',\n            'rateLimit': 1000,\n            'version': 'v2',\n            'hasCORS': false,\n            // obsolete metainfo interface\n            'hasFetchOrder': true,\n            'hasWithdraw': true,\n            // new metainfo interface\n            'has': {\n                'fetchOrder': true,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27786377-8c8ab57e-5fe9-11e7-8ea4-2b05b6bcceec.jpg',\n                'api': 'https://www.bitstamp.net/api',\n                'www': 'https://www.bitstamp.net',\n                'doc': 'https://www.bitstamp.net/api',\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'uid': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'order_book/{pair}/',\n                        'ticker_hour/{pair}/',\n                        'ticker/{pair}/',\n                        'transactions/{pair}/',\n                        'trading-pairs-info/',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balance/',\n                        'balance/{pair}/',\n                        'bch_withdrawal/',\n                        'bch_address/',\n                        'user_transactions/',\n                        'user_transactions/{pair}/',\n                        'open_orders/all/',\n                        'open_orders/{pair}/',\n                        'order_status/',\n                        'cancel_order/',\n                        'buy/{pair}/',\n                        'buy/market/{pair}/',\n                        'sell/{pair}/',\n                        'sell/market/{pair}/',\n                        'ltc_withdrawal/',\n                        'ltc_address/',\n                        'eth_withdrawal/',\n                        'eth_address/',\n                        'xrp_withdrawal/',\n                        'xrp_address/',\n                        'transfer-to-main/',\n                        'transfer-from-main/',\n                        'withdrawal/open/',\n                        'withdrawal/status/',\n                        'withdrawal/cancel/',\n                        'liquidation_address/new/',\n                        'liquidation_address/info/',\n                    ],\n                },\n                'v1': {\n                    'post': [\n                        'bitcoin_deposit_address/',\n                        'unconfirmed_btc/',\n                        'bitcoin_withdrawal/',\n                        'ripple_withdrawal/',\n                        'ripple_address/',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': true,\n                    'percentage': true,\n                    'taker': 0.25 / 100,\n                    'maker': 0.25 / 100,\n                    'tiers': {\n                        'taker': [\n                            [0, 0.25 / 100],\n                            [20000, 0.24 / 100],\n                            [100000, 0.22 / 100],\n                            [400000, 0.20 / 100],\n                            [600000, 0.15 / 100],\n                            [1000000, 0.14 / 100],\n                            [2000000, 0.13 / 100],\n                            [4000000, 0.12 / 100],\n                            [20000000, 0.11 / 100],\n                            [20000001, 0.10 / 100],\n                        ],\n                        'maker': [\n                            [0, 0.25 / 100],\n                            [20000, 0.24 / 100],\n                            [100000, 0.22 / 100],\n                            [400000, 0.20 / 100],\n                            [600000, 0.15 / 100],\n                            [1000000, 0.14 / 100],\n                            [2000000, 0.13 / 100],\n                            [4000000, 0.12 / 100],\n                            [20000000, 0.11 / 100],\n                            [20000001, 0.10 / 100],\n                        ],\n                    },\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0,\n                        'BCH': 0,\n                        'LTC': 0,\n                        'ETH': 0,\n                        'XRP': 0,\n                        'USD': 25,\n                        'EUR': 0.90,\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                        'BCH': 0,\n                        'LTC': 0,\n                        'ETH': 0,\n                        'XRP': 0,\n                        'USD': 25,\n                        'EUR': 0,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetTradingPairsInfo ();\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let symbol = market['name'];\n            let [ base, quote ] = symbol.split ('/');\n            let baseId = base.toLowerCase ();\n            let quoteId = quote.toLowerCase ();\n            let symbolId = baseId + '_' + quoteId;\n            let id = market['url_symbol'];\n            let precision = {\n                'amount': market['base_decimals'],\n                'price': market['counter_decimals'],\n            };\n            let [ cost, currency ] = market['minimum_order'].split (' ');\n            let active = (market['trading'] === 'Enabled');\n            let lot = Math.pow (10, -precision['amount']);\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'baseId': baseId,\n                'quoteId': quoteId,\n                'symbolId': symbolId,\n                'info': market,\n                'lot': lot,\n                'active': active,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision['price']),\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': parseFloat (cost),\n                        'max': undefined,\n                    },\n                },\n            });\n        }\n        return result;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderBookPair (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        let timestamp = parseInt (orderbook['timestamp']) * 1000;\n        return this.parseOrderBook (orderbook, timestamp);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let ticker = await this.publicGetTickerPair (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        let timestamp = parseInt (ticker['timestamp']) * 1000;\n        let vwap = parseFloat (ticker['vwap']);\n        let baseVolume = parseFloat (ticker['volume']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': vwap,\n            'open': parseFloat (ticker['open']),\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = undefined;\n        if ('date' in trade) {\n            timestamp = parseInt (trade['date']) * 1000;\n        } else if ('datetime' in trade) {\n            timestamp = this.parse8601 (trade['datetime']);\n        }\n        let side = (trade['type'] === '0') ? 'buy' : 'sell';\n        let order = undefined;\n        if ('order_id' in trade)\n            order = trade['order_id'].toString ();\n        if ('currency_pair' in trade) {\n            let marketId = trade['currency_pair'];\n            if (marketId in this.markets_by_id)\n                market = this.markets_by_id[marketId];\n        }\n        let price = this.safeFloat (trade, 'price');\n        price = this.safeFloat (trade, market['symbolId'], price);\n        let amount = this.safeFloat (trade, 'amount');\n        amount = this.safeFloat (trade, market['baseId'], amount);\n        let id = this.safeValue (trade, 'tid');\n        id = this.safeValue (trade, 'id', id);\n        if (id)\n            id = id.toString ();\n        return {\n            'id': id,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': order,\n            'type': undefined,\n            'side': side,\n            'price': parseFloat (price),\n            'amount': parseFloat (amount),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTransactionsPair (this.extend ({\n            'pair': market['id'],\n            'time': 'minute',\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balance = await this.privatePostBalance ();\n        let result = { 'info': balance };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let total = lowercase + '_balance';\n            let free = lowercase + '_available';\n            let used = lowercase + '_reserved';\n            let account = this.account ();\n            if (free in balance)\n                account['free'] = parseFloat (balance[free]);\n            if (used in balance)\n                account['used'] = parseFloat (balance[used]);\n            if (total in balance)\n                account['total'] = parseFloat (balance[total]);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let method = 'privatePost' + this.capitalize (side);\n        let order = {\n            'pair': this.marketId (symbol),\n            'amount': amount,\n        };\n        if (type === 'market')\n            method += 'Market';\n        else\n            order['price'] = price;\n        method += 'Pair';\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelOrder ({ 'id': id });\n    }\n\n    parseOrderStatus (order) {\n        if ((order['status'] === 'Queue') || (order['status'] === 'Open'))\n            return 'open';\n        if (order['status'] === 'Finished')\n            return 'closed';\n        return order['status'];\n    }\n\n    async fetchOrderStatus (id, symbol = undefined) {\n        await this.loadMarkets ();\n        let response = await this.privatePostOrderStatus ({ 'id': id });\n        return this.parseOrderStatus (response);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        if (symbol)\n            market = this.market (symbol);\n        let pair = market ? market['id'] : 'all';\n        let request = this.extend ({ 'pair': pair }, params);\n        let response = await this.privatePostUserTransactionsPair (request);\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostOrderStatus ({ 'id': id });\n    }\n\n    getCurrencyName (code) {\n        if (code === 'BTC')\n            return 'bitcoin';\n        return code.toLowerCase ();\n    }\n\n    isFiat (code) {\n        if (code === 'USD')\n            return true;\n        if (code === 'EUR')\n            return true;\n        return false;\n    }\n\n    async withdraw (code, amount, address, tag = undefined, params = {}) {\n        let isFiat = this.isFiat (code);\n        if (isFiat)\n            throw new ExchangeError (this.id + ' fiat withdraw() for ' + code + ' is not implemented yet');\n        let name = this.getCurrencyName (code);\n        let request = {\n            'amount': amount,\n            'address': address,\n        };\n        let v1 = (code === 'BTC');\n        let method = v1 ? 'v1' : 'private'; // v1 or v2\n        method += 'Post' + this.capitalize (name) + 'Withdrawal';\n        let query = params;\n        if (code === 'XRP') {\n            let tag = this.safeString (params, 'destination_tag');\n            if (tag) {\n                request['destination_tag'] = tag;\n                query = this.omit (params, 'destination_tag');\n            } else {\n                throw new ExchangeError (this.id + ' withdraw() requires a destination_tag param for ' + code);\n            }\n        }\n        let response = await this[method] (this.extend (request, query));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/';\n        if (api !== 'v1')\n            url += this.version + '/';\n        url += this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let auth = nonce + this.uid + this.apiKey;\n            let signature = this.encode (this.hmac (this.encode (auth), this.encode (this.secret)));\n            query = this.extend ({\n                'key': this.apiKey,\n                'signature': signature.toUpperCase (),\n                'nonce': nonce,\n            }, query);\n            body = this.urlencode (query);\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('status' in response)\n            if (response['status'] === 'error')\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, NotSupported } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitstamp1 extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitstamp1',\n            'name': 'Bitstamp v1',\n            'countries': 'GB',\n            'rateLimit': 1000,\n            'version': 'v1',\n            'hasCORS': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27786377-8c8ab57e-5fe9-11e7-8ea4-2b05b6bcceec.jpg',\n                'api': 'https://www.bitstamp.net/api',\n                'www': 'https://www.bitstamp.net',\n                'doc': 'https://www.bitstamp.net/api',\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'uid': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'ticker',\n                        'ticker_hour',\n                        'order_book',\n                        'transactions',\n                        'eur_usd',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balance',\n                        'user_transactions',\n                        'open_orders',\n                        'order_status',\n                        'cancel_order',\n                        'cancel_all_orders',\n                        'buy',\n                        'sell',\n                        'bitcoin_deposit_address',\n                        'unconfirmed_btc',\n                        'ripple_withdrawal',\n                        'ripple_address',\n                        'withdrawal_requests',\n                        'bitcoin_withdrawal',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/USD': { 'id': 'btcusd', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD', 'maker': 0.0025, 'taker': 0.0025 },\n                'BTC/EUR': { 'id': 'btceur', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR', 'maker': 0.0025, 'taker': 0.0025 },\n                'EUR/USD': { 'id': 'eurusd', 'symbol': 'EUR/USD', 'base': 'EUR', 'quote': 'USD', 'maker': 0.0025, 'taker': 0.0025 },\n                'XRP/USD': { 'id': 'xrpusd', 'symbol': 'XRP/USD', 'base': 'XRP', 'quote': 'USD', 'maker': 0.0025, 'taker': 0.0025 },\n                'XRP/EUR': { 'id': 'xrpeur', 'symbol': 'XRP/EUR', 'base': 'XRP', 'quote': 'EUR', 'maker': 0.0025, 'taker': 0.0025 },\n                'XRP/BTC': { 'id': 'xrpbtc', 'symbol': 'XRP/BTC', 'base': 'XRP', 'quote': 'BTC', 'maker': 0.0025, 'taker': 0.0025 },\n                'LTC/USD': { 'id': 'ltcusd', 'symbol': 'LTC/USD', 'base': 'LTC', 'quote': 'USD', 'maker': 0.0025, 'taker': 0.0025 },\n                'LTC/EUR': { 'id': 'ltceur', 'symbol': 'LTC/EUR', 'base': 'LTC', 'quote': 'EUR', 'maker': 0.0025, 'taker': 0.0025 },\n                'LTC/BTC': { 'id': 'ltcbtc', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC', 'maker': 0.0025, 'taker': 0.0025 },\n                'ETH/USD': { 'id': 'ethusd', 'symbol': 'ETH/USD', 'base': 'ETH', 'quote': 'USD', 'maker': 0.0025, 'taker': 0.0025 },\n                'ETH/EUR': { 'id': 'etheur', 'symbol': 'ETH/EUR', 'base': 'ETH', 'quote': 'EUR', 'maker': 0.0025, 'taker': 0.0025 },\n                'ETH/BTC': { 'id': 'ethbtc', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC', 'maker': 0.0025, 'taker': 0.0025 },\n            },\n        });\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        if (symbol !== 'BTC/USD')\n            throw new ExchangeError (this.id + ' ' + this.version + \" fetchOrderBook doesn't support \" + symbol + ', use it for BTC/USD only');\n        let orderbook = await this.publicGetOrderBook (params);\n        let timestamp = parseInt (orderbook['timestamp']) * 1000;\n        return this.parseOrderBook (orderbook, timestamp);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        if (symbol !== 'BTC/USD')\n            throw new ExchangeError (this.id + ' ' + this.version + \" fetchTicker doesn't support \" + symbol + ', use it for BTC/USD only');\n        let ticker = await this.publicGetTicker (params);\n        let timestamp = parseInt (ticker['timestamp']) * 1000;\n        let vwap = parseFloat (ticker['vwap']);\n        let baseVolume = parseFloat (ticker['volume']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': vwap,\n            'open': parseFloat (ticker['open']),\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = undefined;\n        if ('date' in trade) {\n            timestamp = parseInt (trade['date']) * 1000;\n        } else if ('datetime' in trade) {\n            // timestamp = this.parse8601 (trade['datetime']);\n            timestamp = parseInt (trade['datetime']) * 1000;\n        }\n        let side = (trade['type'] === 0) ? 'buy' : 'sell';\n        let order = undefined;\n        if ('order_id' in trade)\n            order = trade['order_id'].toString ();\n        if ('currency_pair' in trade) {\n            if (trade['currency_pair'] in this.markets_by_id)\n                market = this.markets_by_id[trade['currency_pair']];\n        }\n        return {\n            'id': trade['tid'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': order,\n            'type': undefined,\n            'side': side,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        if (symbol !== 'BTC/USD')\n            throw new ExchangeError (this.id + ' ' + this.version + \" fetchTrades doesn't support \" + symbol + ', use it for BTC/USD only');\n        let market = this.market (symbol);\n        let response = await this.publicGetTransactions (this.extend ({\n            'time': 'minute',\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        let balance = await this.privatePostBalance ();\n        let result = { 'info': balance };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let total = lowercase + '_balance';\n            let free = lowercase + '_available';\n            let used = lowercase + '_reserved';\n            let account = this.account ();\n            account['free'] = this.safeFloat (balance, free, 0.0);\n            account['used'] = this.safeFloat (balance, used, 0.0);\n            account['total'] = this.safeFloat (balance, total, 0.0);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type !== 'limit')\n            throw new ExchangeError (this.id + ' ' + this.version + ' accepts limit orders only');\n        if (symbol !== 'BTC/USD')\n            throw new ExchangeError (this.id + ' v1 supports BTC/USD orders only');\n        let method = 'privatePost' + this.capitalize (side);\n        let order = {\n            'amount': amount,\n            'price': price,\n        };\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelOrder ({ 'id': id });\n    }\n\n    parseOrderStatus (order) {\n        if ((order['status'] === 'Queue') || (order['status'] === 'Open'))\n            return 'open';\n        if (order['status'] === 'Finished')\n            return 'closed';\n        return order['status'];\n    }\n\n    async fetchOrderStatus (id, symbol = undefined) {\n        await this.loadMarkets ();\n        let response = await this.privatePostOrderStatus ({ 'id': id });\n        return this.parseOrderStatus (response);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        if (symbol)\n            market = this.market (symbol);\n        let pair = market ? market['id'] : 'all';\n        let request = this.extend ({ 'id': pair }, params);\n        let response = await this.privatePostOpenOrdersId (request);\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchOrder is not implemented yet');\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let auth = nonce + this.uid + this.apiKey;\n            let signature = this.encode (this.hmac (this.encode (auth), this.encode (this.secret)));\n            query = this.extend ({\n                'key': this.apiKey,\n                'signature': signature.toUpperCase (),\n                'nonce': nonce,\n            }, query);\n            body = this.urlencode (query);\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('status' in response)\n            if (response['status'] === 'error')\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, AuthenticationError, InvalidOrder, InsufficientFunds, OrderNotFound, DDoSProtection } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bittrex extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bittrex',\n            'name': 'Bittrex',\n            'countries': 'US',\n            'version': 'v1.1',\n            'rateLimit': 1500,\n            'hasAlreadyAuthenticatedSuccessfully': false, // a workaround for APIKEY_INVALID\n            'hasCORS': false,\n            // obsolete metainfo interface\n            'hasFetchTickers': true,\n            'hasFetchOHLCV': true,\n            'hasFetchOrder': true,\n            'hasFetchOrders': true,\n            'hasFetchClosedOrders': true,\n            'hasFetchOpenOrders': true,\n            'hasFetchMyTrades': false,\n            'hasFetchCurrencies': true,\n            'hasWithdraw': true,\n            // new metainfo interface\n            'has': {\n                'fetchTickers': true,\n                'fetchOHLCV': true,\n                'fetchOrder': true,\n                'fetchOrders': true,\n                'fetchClosedOrders': 'emulated',\n                'fetchOpenOrders': true,\n                'fetchMyTrades': false,\n                'fetchCurrencies': true,\n                'withdraw': true,\n            },\n            'timeframes': {\n                '1m': 'oneMin',\n                '5m': 'fiveMin',\n                '30m': 'thirtyMin',\n                '1h': 'hour',\n                '1d': 'day',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766352-cf0b3c26-5ed5-11e7-82b7-f3826b7a97d8.jpg',\n                'api': {\n                    'public': 'https://bittrex.com/api',\n                    'account': 'https://bittrex.com/api',\n                    'market': 'https://bittrex.com/api',\n                    'v2': 'https://bittrex.com/api/v2.0/pub',\n                },\n                'www': 'https://bittrex.com',\n                'doc': [\n                    'https://bittrex.com/Home/Api',\n                    'https://www.npmjs.org/package/node.bittrex.api',\n                ],\n                'fees': [\n                    'https://bittrex.com/Fees',\n                    'https://support.bittrex.com/hc/en-us/articles/115000199651-What-fees-does-Bittrex-charge-',\n                ],\n            },\n            'api': {\n                'v2': {\n                    'get': [\n                        'currencies/GetBTCPrice',\n                        'market/GetTicks',\n                        'market/GetLatestTick',\n                        'Markets/GetMarketSummaries',\n                        'market/GetLatestTick',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'currencies',\n                        'markethistory',\n                        'markets',\n                        'marketsummaries',\n                        'marketsummary',\n                        'orderbook',\n                        'ticker',\n                    ],\n                },\n                'account': {\n                    'get': [\n                        'balance',\n                        'balances',\n                        'depositaddress',\n                        'deposithistory',\n                        'order',\n                        'orderhistory',\n                        'withdrawalhistory',\n                        'withdraw',\n                    ],\n                },\n                'market': {\n                    'get': [\n                        'buylimit',\n                        'buymarket',\n                        'cancel',\n                        'openorders',\n                        'selllimit',\n                        'sellmarket',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'maker': 0.0025,\n                    'taker': 0.0025,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.001,\n                        'LTC': 0.01,\n                        'DOGE': 2,\n                        'VTC': 0.02,\n                        'PPC': 0.02,\n                        'FTC': 0.2,\n                        'RDD': 2,\n                        'NXT': 2,\n                        'DASH': 0.002,\n                        'POT': 0.002,\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                        'LTC': 0,\n                        'DOGE': 0,\n                        'VTC': 0,\n                        'PPC': 0,\n                        'FTC': 0,\n                        'RDD': 0,\n                        'NXT': 0,\n                        'DASH': 0,\n                        'POT': 0,\n                    },\n                },\n            },\n        });\n    }\n\n    costToPrecision (symbol, cost) {\n        return this.truncate (parseFloat (cost), this.markets[symbol]['precision']['price']);\n    }\n\n    feeToPrecision (symbol, fee) {\n        return this.truncate (parseFloat (fee), this.markets[symbol]['precision']['price']);\n    }\n\n    async fetchMarkets () {\n        let response = await this.v2GetMarketsGetMarketSummaries ();\n        let result = [];\n        for (let i = 0; i < response['result'].length; i++) {\n            let market = response['result'][i]['Market'];\n            let id = market['MarketName'];\n            let base = market['MarketCurrency'];\n            let quote = market['BaseCurrency'];\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            let precision = {\n                'amount': 8,\n                'price': 8,\n            };\n            let active = market['IsActive'];\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': active,\n                'info': market,\n                'lot': Math.pow (10, -precision['amount']),\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': market['MinTradeSize'],\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                },\n            }));\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.accountGetBalances ();\n        let balances = response['result'];\n        let result = { 'info': balances };\n        let indexed = this.indexBy (balances, 'Currency');\n        let keys = Object.keys (indexed);\n        for (let i = 0; i < keys.length; i++) {\n            let id = keys[i];\n            let currency = this.commonCurrencyCode (id);\n            let account = this.account ();\n            let balance = indexed[id];\n            let free = parseFloat (balance['Available']);\n            let total = parseFloat (balance['Balance']);\n            let used = total - free;\n            account['free'] = free;\n            account['used'] = used;\n            account['total'] = total;\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetOrderbook (this.extend ({\n            'market': this.marketId (symbol),\n            'type': 'both',\n        }, params));\n        let orderbook = response['result'];\n        if ('type' in params) {\n            if (params['type'] === 'buy') {\n                orderbook = {\n                    'buy': response['result'],\n                    'sell': [],\n                };\n            } else if (params['type'] === 'sell') {\n                orderbook = {\n                    'buy': [],\n                    'sell': response['result'],\n                };\n            }\n        }\n        return this.parseOrderBook (orderbook, undefined, 'buy', 'sell', 'Rate', 'Quantity');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.parse8601 (ticker['TimeStamp']);\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'High'),\n            'low': this.safeFloat (ticker, 'Low'),\n            'bid': this.safeFloat (ticker, 'Bid'),\n            'ask': this.safeFloat (ticker, 'Ask'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'Last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'Volume'),\n            'quoteVolume': this.safeFloat (ticker, 'BaseVolume'),\n            'info': ticker,\n        };\n    }\n\n    async fetchCurrencies (params = {}) {\n        let response = await this.publicGetCurrencies (params);\n        let currencies = response['result'];\n        let result = {};\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let id = currency['Currency'];\n            // todo: will need to rethink the fees\n            // to add support for multiple withdrawal/deposit methods and\n            // differentiated fees for each particular method\n            let code = this.commonCurrencyCode (id);\n            let precision = 8; // default precision, todo: fix \"magic constants\"\n            result[code] = {\n                'id': id,\n                'code': code,\n                'info': currency,\n                'name': currency['CurrencyLong'],\n                'active': currency['IsActive'],\n                'status': 'ok',\n                'fee': currency['TxFee'], // todo: redesign\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': currency['TxFee'],\n                        'max': Math.pow (10, precision),\n                    },\n                },\n            };\n        }\n        return result;\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetMarketsummaries (params);\n        let tickers = response['result'];\n        let result = {};\n        for (let t = 0; t < tickers.length; t++) {\n            let ticker = tickers[t];\n            let id = ticker['MarketName'];\n            let market = undefined;\n            let symbol = id;\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            } else {\n                symbol = this.parseSymbol (id);\n            }\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetMarketsummary (this.extend ({\n            'market': market['id'],\n        }, params));\n        let ticker = response['result'][0];\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = this.parse8601 (trade['TimeStamp']);\n        let side = undefined;\n        if (trade['OrderType'] === 'BUY') {\n            side = 'buy';\n        } else if (trade['OrderType'] === 'SELL') {\n            side = 'sell';\n        }\n        let id = undefined;\n        if ('Id' in trade)\n            id = trade['Id'].toString ();\n        return {\n            'id': id,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': 'limit',\n            'side': side,\n            'price': parseFloat (trade['Price']),\n            'amount': parseFloat (trade['Quantity']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetMarkethistory (this.extend ({\n            'market': market['id'],\n        }, params));\n        if ('result' in response) {\n            if (typeof response['result'] !== 'undefined')\n                return this.parseTrades (response['result'], market, since, limit);\n        }\n        throw new ExchangeError (this.id + ' fetchTrades() returned undefined response');\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1d', since = undefined, limit = undefined) {\n        let timestamp = this.parse8601 (ohlcv['T']);\n        return [\n            timestamp,\n            ohlcv['O'],\n            ohlcv['H'],\n            ohlcv['L'],\n            ohlcv['C'],\n            ohlcv['V'],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'tickInterval': this.timeframes[timeframe],\n            'marketName': market['id'],\n        };\n        let response = await this.v2GetMarketGetTicks (this.extend (request, params));\n        if ('result' in response) {\n            if (response['result'])\n                return this.parseOHLCVs (response['result'], market, timeframe, since, limit);\n        }\n        throw new ExchangeError (this.id + ' returned an empty or unrecognized response: ' + this.json (response));\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['market'] = market['id'];\n        }\n        let response = await this.marketGetOpenorders (this.extend (request, params));\n        let orders = this.parseOrders (response['result'], market, since, limit);\n        return this.filterOrdersBySymbol (orders, symbol);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type !== 'limit')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'marketGet' + this.capitalize (side) + type;\n        let order = {\n            'market': market['id'],\n            'quantity': this.amountToPrecision (symbol, amount),\n            'rate': this.priceToPrecision (symbol, price),\n        };\n        // if (type == 'limit')\n        //     order['rate'] = this.priceToPrecision (symbol, price);\n        let response = await this[method] (this.extend (order, params));\n        let orderIdField = this.getOrderIdField ();\n        let result = {\n            'info': response,\n            'id': response['result'][orderIdField],\n        };\n        return result;\n    }\n\n    getOrderIdField () {\n        return 'uuid';\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = undefined;\n        try {\n            let orderIdField = this.getOrderIdField ();\n            let request = {};\n            request[orderIdField] = id;\n            response = await this.marketGetCancel (this.extend (request, params));\n        } catch (e) {\n            if (this.last_json_response) {\n                let message = this.safeString (this.last_json_response, 'message');\n                if (message === 'ORDER_NOT_OPEN')\n                    throw new InvalidOrder (this.id + ' cancelOrder() error: ' + this.last_http_response);\n                if (message === 'UUID_INVALID')\n                    throw new OrderNotFound (this.id + ' cancelOrder() error: ' + this.last_http_response);\n            }\n            throw e;\n        }\n        return response;\n    }\n\n    parseSymbol (id) {\n        let [ quote, base ] = id.split ('-');\n        base = this.commonCurrencyCode (base);\n        quote = this.commonCurrencyCode (quote);\n        return base + '/' + quote;\n    }\n\n    parseOrder (order, market = undefined) {\n        let side = this.safeString (order, 'OrderType');\n        if (typeof side === 'undefined')\n            side = this.safeString (order, 'Type');\n        let isBuyOrder = (side === 'LIMIT_BUY') || (side === 'BUY');\n        side = isBuyOrder ? 'buy' : 'sell';\n        let status = 'open';\n        if (('Closed' in order) && order['Closed']) {\n            status = 'closed';\n        } else if (('CancelInitiated' in order) && order['CancelInitiated']) {\n            status = 'canceled';\n        }\n        let symbol = undefined;\n        if (!market) {\n            if ('Exchange' in order) {\n                let marketId = order['Exchange'];\n                if (marketId in this.markets_by_id)\n                    market = this.markets_by_id[marketId];\n                else\n                    symbol = this.parseSymbol (marketId);\n            }\n        }\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = undefined;\n        if ('Opened' in order)\n            timestamp = this.parse8601 (order['Opened']);\n        if ('TimeStamp' in order)\n            timestamp = this.parse8601 (order['TimeStamp']);\n        if ('Created' in order)\n            timestamp = this.parse8601 (order['Created']);\n        let fee = undefined;\n        let commission = undefined;\n        if ('Commission' in order) {\n            commission = 'Commission';\n        } else if ('CommissionPaid' in order) {\n            commission = 'CommissionPaid';\n        }\n        if (commission) {\n            fee = {\n                'cost': parseFloat (order[commission]),\n            };\n            if (market)\n                fee['currency'] = market['quote'];\n        }\n        let price = this.safeFloat (order, 'Limit');\n        let cost = this.safeFloat (order, 'Price');\n        let amount = this.safeFloat (order, 'Quantity');\n        let remaining = this.safeFloat (order, 'QuantityRemaining', 0.0);\n        let filled = amount - remaining;\n        if (!cost) {\n            if (price && amount)\n                cost = price * amount;\n        }\n        if (!price) {\n            if (cost && filled)\n                price = cost / filled;\n        }\n        let average = this.safeFloat (order, 'PricePerUnit');\n        let id = this.safeString (order, 'OrderUuid');\n        if (typeof id === 'undefined')\n            id = this.safeString (order, 'OrderId');\n        let result = {\n            'info': order,\n            'id': id,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': 'limit',\n            'side': side,\n            'price': price,\n            'cost': cost,\n            'average': average,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': fee,\n        };\n        return result;\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = undefined;\n        try {\n            let orderIdField = this.getOrderIdField ();\n            let request = {};\n            request[orderIdField] = id;\n            response = await this.accountGetOrder (this.extend (request, params));\n        } catch (e) {\n            if (this.last_json_response) {\n                let message = this.safeString (this.last_json_response, 'message');\n                if (message === 'UUID_INVALID')\n                    throw new OrderNotFound (this.id + ' fetchOrder() error: ' + this.last_http_response);\n            }\n            throw e;\n        }\n        return this.parseOrder (response['result']);\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['market'] = market['id'];\n        }\n        let response = await this.accountGetOrderhistory (this.extend (request, params));\n        let orders = this.parseOrders (response['result'], market, since, limit);\n        if (symbol)\n            return this.filterOrdersBySymbol (orders, symbol);\n        return orders;\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, since, limit, params);\n        return this.filterBy (orders, 'status', 'closed');\n    }\n\n    currencyId (currency) {\n        if (currency === 'BCH')\n            return 'BCC';\n        return currency;\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        let currencyId = this.currencyId (currency);\n        let response = await this.accountGetDepositaddress (this.extend ({\n            'currency': currencyId,\n        }, params));\n        let address = this.safeString (response['result'], 'Address');\n        let message = this.safeString (response, 'message');\n        let status = 'ok';\n        if (!address || message === 'ADDRESS_GENERATING')\n            status = 'pending';\n        return {\n            'currency': currency,\n            'address': address,\n            'status': status,\n            'info': response,\n        };\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        let currencyId = this.currencyId (currency);\n        let request = {\n            'currency': currencyId,\n            'quantity': amount,\n            'address': address,\n        };\n        if (tag)\n            request['paymentid'] = tag;\n        let response = await this.accountGetWithdraw (this.extend (request, params));\n        let id = undefined;\n        if ('result' in response) {\n            if ('uuid' in response['result'])\n                id = response['result']['uuid'];\n        }\n        return {\n            'info': response,\n            'id': id,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api] + '/';\n        if (api !== 'v2')\n            url += this.version + '/';\n        if (api === 'public') {\n            url += api + '/' + method.toLowerCase () + path;\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else if (api === 'v2') {\n            url += path;\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            url += api + '/';\n            if (((api === 'account') && (path !== 'withdraw')) || (path === 'openorders'))\n                url += method.toLowerCase ();\n            url += path + '?' + this.urlencode (this.extend ({\n                'nonce': nonce,\n                'apikey': this.apiKey,\n            }, params));\n            let signature = this.hmac (this.encode (url), this.encode (this.secret), 'sha512');\n            headers = { 'apisign': signature };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    throwExceptionOnError (response) {\n        if ('message' in response) {\n            if (response['message'] === 'APISIGN_NOT_PROVIDED')\n                throw new AuthenticationError (this.id + ' ' + this.json (response));\n            if (response['message'] === 'INVALID_SIGNATURE')\n                throw new AuthenticationError (this.id + ' ' + this.json (response));\n            if (response['message'] === 'INSUFFICIENT_FUNDS')\n                throw new InsufficientFunds (this.id + ' ' + this.json (response));\n            if (response['message'] === 'MIN_TRADE_REQUIREMENT_NOT_MET')\n                throw new InvalidOrder (this.id + ' ' + this.json (response));\n            if (response['message'] === 'APIKEY_INVALID') {\n                if (this.hasAlreadyAuthenticatedSuccessfully) {\n                    throw new DDoSProtection (this.id + ' ' + this.json (response));\n                } else {\n                    throw new AuthenticationError (this.id + ' ' + this.json (response));\n                }\n            }\n            if (response['message'] === 'DUST_TRADE_DISALLOWED_MIN_VALUE_50K_SAT')\n                throw new InvalidOrder (this.id + ' order cost should be over 50k satoshi ' + this.json (response));\n        }\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code >= 400) {\n            if (body[0] === '{') {\n                let response = JSON.parse (body);\n                this.throwExceptionOnError (response);\n                if ('success' in response) {\n                    let success = response['success'];\n                    if (typeof success === 'string')\n                        success = (success === 'true') ? true : false;\n                    if (!success) {\n                        this.throwExceptionOnError (response);\n                        throw new ExchangeError (this.id + ' ' + this.json (response));\n                    }\n                }\n            }\n        }\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('success' in response) {\n            let success = response['success'];\n            if (typeof success === 'string')\n                success = (success === 'true') ? true : false;\n            if (success) {\n                // a workaround for APIKEY_INVALID\n                if ((api === 'account') || (api === 'market'))\n                    this.hasAlreadyAuthenticatedSuccessfully = true;\n                return response;\n            }\n        }\n        this.throwExceptionOnError (response);\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class bl3p extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bl3p',\n            'name': 'BL3P',\n            'countries': [ 'NL', 'EU' ], // Netherlands, EU\n            'rateLimit': 1000,\n            'version': '1',\n            'comment': 'An exchange market by BitonicNL',\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28501752-60c21b82-6feb-11e7-818b-055ee6d0e754.jpg',\n                'api': 'https://api.bl3p.eu',\n                'www': [\n                    'https://bl3p.eu',\n                    'https://bitonic.nl',\n                ],\n                'doc': [\n                    'https://github.com/BitonicNL/bl3p-api/tree/master/docs',\n                    'https://bl3p.eu/api',\n                    'https://bitonic.nl/en/api',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        '{market}/ticker',\n                        '{market}/orderbook',\n                        '{market}/trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        '{market}/money/depth/full',\n                        '{market}/money/order/add',\n                        '{market}/money/order/cancel',\n                        '{market}/money/order/result',\n                        '{market}/money/orders',\n                        '{market}/money/orders/history',\n                        '{market}/money/trades/fetch',\n                        'GENMKT/money/info',\n                        'GENMKT/money/deposit_address',\n                        'GENMKT/money/new_deposit_address',\n                        'GENMKT/money/wallet/history',\n                        'GENMKT/money/withdraw',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/EUR': { 'id': 'BTCEUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR', 'maker': 0.0025, 'taker': 0.0025 },\n                'LTC/EUR': { 'id': 'LTCEUR', 'symbol': 'LTC/EUR', 'base': 'LTC', 'quote': 'EUR', 'maker': 0.0025, 'taker': 0.0025 },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostGENMKTMoneyInfo ();\n        let data = response['data'];\n        let balance = data['wallets'];\n        let result = { 'info': data };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            if (currency in balance) {\n                if ('available' in balance[currency]) {\n                    account['free'] = parseFloat (balance[currency]['available']['value']);\n                }\n            }\n            if (currency in balance) {\n                if ('balance' in balance[currency]) {\n                    account['total'] = parseFloat (balance[currency]['balance']['value']);\n                }\n            }\n            if (account['total']) {\n                if (account['free']) {\n                    account['used'] = account['total'] - account['free'];\n                }\n            }\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    parseBidAsk (bidask, priceKey = 0, amountKey = 0) {\n        return [\n            bidask['price_int'] / 100000.0,\n            bidask['amount_int'] / 100000000.0,\n        ];\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetMarketOrderbook (this.extend ({\n            'market': market['id'],\n        }, params));\n        let orderbook = response['data'];\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetMarketTicker (this.extend ({\n            'market': this.marketId (symbol),\n        }, params));\n        let timestamp = ticker['timestamp'] * 1000;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume']['24h']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        return {\n            'id': trade['trade_id'].toString (),\n            'timestamp': trade['date'],\n            'datetime': this.iso8601 (trade['date']),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': trade['price_int'] / 100000.0,\n            'amount': trade['amount_int'] / 100000000.0,\n            'info': trade,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetMarketTrades (this.extend ({\n            'market': market['id'],\n        }, params));\n        let result = this.parseTrades (response['data']['trades'], market, since, limit);\n        return result;\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let market = this.market (symbol);\n        let order = {\n            'market': market['id'],\n            'amount_int': parseInt (amount * 100000000),\n            'fee_currency': market['quote'],\n            'type': (side == 'buy') ? 'bid' : 'ask',\n        };\n        if (type == 'limit')\n            order['price_int'] = parseInt (price * 100000.0);\n        let response = await this.privatePostMarketMoneyOrderAdd (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['data']['order_id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostMarketMoneyOrderCancel ({ 'order_id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let request = this.implodeParams (path, params);\n        let url = this.urls['api'] + '/' + this.version + '/' + request;\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({ 'nonce': nonce }, query));\n            let secret = this.base64ToBinary (this.secret);\n            let auth = request + \"\\0\" + body;\n            let signature = this.hmac (this.encode (auth), secret, 'sha512', 'base64');\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Rest-Key': this.apiKey,\n                'Rest-Sign': this.decode (signature),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst bittrex = require ('./bittrex.js');\nconst { AuthenticationError, InvalidOrder, InsufficientFunds, DDoSProtection } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class bleutrade extends bittrex {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bleutrade',\n            'name': 'Bleutrade',\n            'countries': 'BR', // Brazil\n            'rateLimit': 1000,\n            'version': 'v2',\n            'hasCORS': true,\n            'hasFetchTickers': true,\n            'hasFetchOHLCV': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30303000-b602dbe6-976d-11e7-956d-36c5049c01e7.jpg',\n                'api': {\n                    'public': 'https://bleutrade.com/api',\n                    'account': 'https://bleutrade.com/api',\n                    'market': 'https://bleutrade.com/api',\n                },\n                'www': 'https://bleutrade.com',\n                'doc': 'https://bleutrade.com/help/API',\n                'fees': 'https://bleutrade.com/help/fees_and_deadlines',\n            },\n            'fees': {\n                'funding': {\n                    'ADC': 0.1,\n                    'BTA': 0.1,\n                    'BITB': 0.1,\n                    'BTC': 0.001,\n                    'BCH': 0.001,\n                    'BTCD': 0.001,\n                    'BTG': 0.001,\n                    'BLK': 0.1,\n                    'CDN': 0.1,\n                    'CLAM': 0.01,\n                    'DASH': 0.001,\n                    'DCR': 0.05,\n                    'DGC': 0.1,\n                    'DP': 0.1,\n                    'DPC': 0.1,\n                    'DOGE': 0.0,\n                    'EFL': 0.1,\n                    'ETH': 0.01,\n                    'EXP': 0.1,\n                    'FJC': 0.1,\n                    'BSTY': 0.001,\n                    'GB': 0.1,\n                    'NLG': 0.1,\n                    'HTML': 1.0,\n                    'LTC': 0.001,\n                    'MONA': 0.01,\n                    'MOON': 1.0,\n                    'NMC': 0.015,\n                    'NEOS': 0.1,\n                    'NVC': 0.05,\n                    'OK': 0.1,\n                    'PPC': 0.1,\n                    'POT': 0.1,\n                    'XPM': 0.001,\n                    'QTUM': 0.1,\n                    'RDD': 0.1,\n                    'SLR': 0.1,\n                    'START': 0.1,\n                    'SLG': 0.1,\n                    'TROLL': 0.1,\n                    'UNO': 0.01,\n                    'VRC': 0.1,\n                    'VTC': 0.1,\n                    'XVP': 0.1,\n                    'WDC': 0.001,\n                    'ZET': 0.1,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetMarkets ();\n        let result = [];\n        for (let p = 0; p < markets['result'].length; p++) {\n            let market = markets['result'][p];\n            let id = market['MarketName'];\n            let base = market['MarketCurrency'];\n            let quote = market['BaseCurrency'];\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            let precision = {\n                'amount': 8,\n                'price': 8,\n            };\n            let active = market['IsActive'];\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': active,\n                'info': market,\n                'lot': Math.pow (10, -precision['amount']),\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': market['MinTradeSize'],\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': 0,\n                        'max': undefined,\n                    },\n                },\n            }));\n        }\n        return result;\n    }\n\n    getOrderIdField () {\n        return 'orderid';\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetOrderbook (this.extend ({\n            'market': this.marketId (symbol),\n            'type': 'ALL',\n            'depth': 50,\n        }, params));\n        let orderbook = response['result'];\n        return this.parseOrderBook (orderbook, undefined, 'buy', 'sell', 'Rate', 'Quantity');\n    }\n\n    throwExceptionOnError (response) {\n        if ('message' in response) {\n            if (response['message'] === 'Insufficient funds!')\n                throw new InsufficientFunds (this.id + ' ' + this.json (response));\n            if (response['message'] === 'MIN_TRADE_REQUIREMENT_NOT_MET')\n                throw new InvalidOrder (this.id + ' ' + this.json (response));\n            if (response['message'] === 'APIKEY_INVALID') {\n                if (this.hasAlreadyAuthenticatedSuccessfully) {\n                    throw new DDoSProtection (this.id + ' ' + this.json (response));\n                } else {\n                    throw new AuthenticationError (this.id + ' ' + this.json (response));\n                }\n            }\n            if (response['message'] === 'DUST_TRADE_DISALLOWED_MIN_VALUE_50K_SAT')\n                throw new InvalidOrder (this.id + ' order cost should be over 50k satoshi ' + this.json (response));\n        }\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InvalidOrder, AuthenticationError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class braziliex extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'braziliex',\n            'name': 'Braziliex',\n            'countries': 'BR',\n            'rateLimit': 1000,\n            // obsolete metainfo interface\n            'hasFetchTickers': true,\n            'hasFetchOpenOrders': true,\n            'hasFetchMyTrades': true,\n            // new metainfo interface\n            'has': {\n                'fetchTickers': true,\n                'fetchOpenOrders': true,\n                'fetchMyTrades': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/34703593-c4498674-f504-11e7-8d14-ff8e44fb78c1.jpg',\n                'api': 'https://braziliex.com/api/v1',\n                'www': 'https://braziliex.com/',\n                'doc': 'https://braziliex.com/exchange/api.php',\n                'fees': 'https://braziliex.com/exchange/fees.php',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'currencies',\n                        'ticker',\n                        'ticker/{market}',\n                        'orderbook/{market}',\n                        'tradehistory/{market}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balance',\n                        'complete_balance',\n                        'open_orders',\n                        'trade_history',\n                        'deposit_address',\n                        'sell',\n                        'buy',\n                        'cancel_order',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.005,\n                    'taker': 0.005,\n                },\n            },\n            'precision': {\n                'amount': 8,\n                'price': 8,\n            },\n        });\n    }\n\n    async fetchCurrencies (params = {}) {\n        let currencies = await this.publicGetCurrencies (params);\n        let ids = Object.keys (currencies);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let currency = currencies[id];\n            let precision = currency['decimal'];\n            let uppercase = id.toUpperCase ();\n            let code = this.commonCurrencyCode (uppercase);\n            let active = currency['active'] == 1;\n            let status = 'ok';\n            if (currency['under_maintenance'] != 0) {\n                active = false;\n                status = 'maintenance';\n            }\n            let canWithdraw = currency['is_withdrawal_active'] == 1;\n            let canDeposit = currency['is_deposit_active'] == 1;\n            if (!canWithdraw || !canDeposit)\n                active = false;\n            result[code] = {\n                'id': id,\n                'code': code,\n                'name': currency['name'],\n                'active': active,\n                'status': status,\n                'precision': precision,\n                'wallet': {\n                    'address': undefined,\n                    'extra': undefined,\n                    'withdraw': {\n                        'active': canWithdraw,\n                        'fee': currency['txWithdrawalFee'],\n                    },\n                    'deposit': {\n                        'active': canDeposit,\n                        'fee': currency['txDepositFee'],\n                    },\n                },\n                'limits': {\n                    'amount': {\n                        'min': currency['minAmountTrade'],\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': currency['MinWithdrawal'],\n                        'max': Math.pow (10, precision),\n                    },\n                    'deposit': {\n                        'min': currency['minDeposit'],\n                        'max': undefined,\n                    },\n                },\n                'info': currency,\n            };\n        }\n        return result;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetTicker ();\n        let ids = Object.keys (markets);\n        let result = [];\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = markets[id];\n            let idUpperCase = id.toUpperCase ();\n            let [ base, quote ] = idUpperCase.split ('_');\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            let active = market['active'] == 1;\n            let precision = {\n                'amount': 8,\n                'price': 8,\n            };\n            let lot = Math.pow (10, -precision['amount']);\n            result.push ({\n                'id': id,\n                'symbol': symbol.toUpperCase (),\n                'base': base,\n                'quote': quote,\n                'active': active,\n                'lot': lot,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': Math.pow (10, precision['amount']),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision['price']),\n                        'max': Math.pow (10, precision['price']),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                },\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let symbol = market['symbol'];\n        let timestamp = ticker['date'];\n        ticker = ticker['ticker'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['highestBid24']),\n            'low': parseFloat (ticker['lowestAsk24']),\n            'bid': parseFloat (ticker['highestBid']),\n            'ask': parseFloat (ticker['lowestAsk']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': parseFloat (ticker['percentChange']),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['baseVolume24']),\n            'quoteVolume': parseFloat (ticker['quoteVolume24']),\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetTickerMarket (this.extend ({\n            'market': market['id'],\n        }, params));\n        ticker = {\n            'date': this.milliseconds (),\n            'ticker': ticker,\n        };\n        return this.parseTicker (ticker, market);\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTicker (params);\n        let result = {};\n        let timestamp = this.milliseconds ();\n        let ids = Object.keys (tickers);\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = {\n                'date': timestamp,\n                'ticker': tickers[id],\n            };\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderbookMarket (this.extend ({\n            'market': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'amount');\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = undefined;\n        if ('date_exec' in trade) {\n            timestamp = this.parse8601 (trade['date_exec']);\n        } else {\n            timestamp = this.parse8601 (trade['date']);\n        }\n        let price = parseFloat (trade['price']);\n        let amount = parseFloat (trade['amount']);\n        let symbol = market['symbol'];\n        let cost = parseFloat (trade['total']);\n        let orderId = this.safeString (trade, 'order_number');\n        return {\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'id': this.safeString (trade, '_id'),\n            'order': orderId,\n            'type': 'limit',\n            'side': trade['type'],\n            'price': price,\n            'amount': amount,\n            'cost': cost,\n            'fee': undefined,\n            'info': trade,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let trades = await this.publicGetTradehistoryMarket (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseTrades (trades, market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privatePostCompleteBalance (params);\n        let result = { 'info': balances };\n        let currencies = Object.keys (balances);\n        for (let i = 0; i < currencies.length; i++) {\n            let id = currencies[i];\n            let balance = balances[id];\n            let currency = this.commonCurrencyCode (id);\n            let account = {\n                'free': parseFloat (balance['available']),\n                'used': 0.0,\n                'total': parseFloat (balance['total']),\n            };\n            account['used'] = account['total'] - account['free'];\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    parseOrder (order, market = undefined) {\n        let symbol = undefined;\n        if (!market) {\n            let marketId = this.safeString (order, 'market');\n            if (marketId)\n                if (marketId in this.markets_by_id)\n                    market = this.markets_by_id[marketId];\n        }\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = this.safeValue (order, 'timestamp');\n        if (!timestamp)\n            timestamp = this.parse8601 (order['date']);\n        let price = parseFloat (order['price']);\n        let cost = this.safeFloat (order, 'total', 0.0);\n        let amount = this.safeFloat (order, 'amount');\n        let filledPercentage = this.safeFloat (order, 'progress');\n        let filled = amount * filledPercentage;\n        let remaining = this.amountToPrecision (symbol, amount - filled);\n        let info = order;\n        if ('info' in info)\n            info = order['info'];\n        return {\n            'id': order['order_number'],\n            'datetime': this.iso8601 (timestamp),\n            'timestamp': timestamp,\n            'status': 'open',\n            'symbol': symbol,\n            'type': 'limit',\n            'side': order['type'],\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'trades': undefined,\n            'fee': this.safeValue (order, 'fee'),\n            'info': info,\n        };\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'privatePost' + this.capitalize (side);\n        let response = await this[method] (this.extend ({\n            'market': market['id'],\n            // 'price': this.priceToPrecision (symbol, price),\n            // 'amount': this.amountToPrecision (symbol, amount),\n            'price': price,\n            'amount': amount,\n        }, params));\n        let success = this.safeInteger (response, 'success');\n        if (success != 1)\n            throw new InvalidOrder (this.id + ' ' + this.json (response));\n        let parts = response['message'].split (' / ');\n        parts = parts.slice (1);\n        let feeParts = parts[5].split (' ');\n        let order = this.parseOrder ({\n            'timestamp': this.milliseconds (),\n            'order_number': response['order_number'],\n            'type': parts[0].toLowerCase (),\n            'market': parts[0].toLowerCase (),\n            'amount': parts[2].split (' ')[1],\n            'price': parts[3].split (' ')[1],\n            'total': parts[4].split (' ')[1],\n            'fee': {\n                'cost': parseFloat (feeParts[1]),\n                'currency': feeParts[2],\n            },\n            'progress': '0.0',\n            'info': response,\n        }, market);\n        let id = order['id'];\n        this.orders[id] = order;\n        return order;\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let result = await this.privatePostCancelOrder (this.extend ({\n            'order_number': id,\n            'market': market['id'],\n        }, params));\n        return result;\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orders = await this.privatePostOpenOrders (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseOrders (orders['order_open'], market, since, limit);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let trades = await this.privatePostTradeHistory (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseTrades (trades['trade_history'], market, since, limit);\n    }\n\n    async fetchDepositAddress (currencyCode, params = {}) {\n        await this.loadMarkets ();\n        let currency = this.currency (currencyCode);\n        let response = await this.privatePostDepositAddress (this.extend ({\n            'currency': currency['id'],\n        }, params));\n        let address = this.safeString (response['deposit_address'], 'address');\n        if (!address)\n            throw new ExchangeError (this.id + ' fetchDepositAddress failed: ' + this.last_http_response);\n        return {\n            'currency': currencyCode,\n            'address': address,\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + api;\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            url += '/' + this.implodeParams (path, params);\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            query = this.extend ({\n                'command': path,\n                'nonce': this.nonce (),\n            }, query);\n            body = this.urlencode (query);\n            let signature = this.hmac (this.encode (body), this.encode (this.secret), 'sha512');\n            headers = {\n                'Content-type': 'application/x-www-form-urlencoded',\n                'Key': this.apiKey,\n                'Sign': this.decode (signature),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('success' in response) {\n            let success = this.safeInteger (response, 'success');\n            if (success == 0) {\n                let message = this.safeString (response, 'message');\n                if (message == 'Invalid APIKey')\n                    throw new AuthenticationError (message);\n                throw new ExchangeError (message);\n            }\n        }\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class btcbox extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'btcbox',\n            'name': 'BtcBox',\n            'countries': 'JP',\n            'rateLimit': 1000,\n            'version': 'v1',\n            'hasCORS': false,\n            'hasFetchOHLCV': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/31275803-4df755a8-aaa1-11e7-9abb-11ec2fad9f2d.jpg',\n                'api': 'https://www.btcbox.co.jp/api',\n                'www': 'https://www.btcbox.co.jp/',\n                'doc': 'https://www.btcbox.co.jp/help/asm',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'depth',\n                        'orders',\n                        'ticker',\n                        'allticker',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balance',\n                        'trade_add',\n                        'trade_cancel',\n                        'trade_list',\n                        'trade_view',\n                        'wallet',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/JPY': { 'id': 'BTC/JPY', 'symbol': 'BTC/JPY', 'base': 'BTC', 'quote': 'JPY' },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privatePostBalance ();\n        let result = { 'info': balances };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            if (lowercase == 'dash')\n                lowercase = 'drk';\n            let account = this.account ();\n            let free = lowercase + '_balance';\n            let used = lowercase + '_lock';\n            if (free in balances)\n                account['free'] = parseFloat (balances[free]);\n            if (used in balances)\n                account['used'] = parseFloat (balances[used]);\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {};\n        let numSymbols = this.symbols.length;\n        if (numSymbols > 1)\n            request['coin'] = market['id'];\n        let orderbook = await this.publicGetDepth (this.extend (request, params));\n        let result = this.parseOrderBook (orderbook);\n        result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'buy'),\n            'ask': this.safeFloat (ticker, 'sell'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'vol'),\n            'quoteVolume': this.safeFloat (ticker, 'volume'),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetAllticker (params);\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {};\n        let numSymbols = this.symbols.length;\n        if (numSymbols > 1)\n            request['coin'] = market['id'];\n        let ticker = await this.publicGetTicker (this.extend (request, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        return {\n            'info': trade,\n            'id': trade['tid'],\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['type'],\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {};\n        let numSymbols = this.symbols.length;\n        if (numSymbols > 1)\n            request['coin'] = market['id'];\n        let response = await this.publicGetOrders (this.extend (request, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'amount': amount,\n            'price': price,\n            'type': side,\n        };\n        let numSymbols = this.symbols.length;\n        if (numSymbols > 1)\n            request['coin'] = market['id'];\n        let response = await this.privatePostTradeAdd (this.extend (request, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostTradeCancel (this.extend ({\n            'id': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + path;\n        if (api == 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let query = this.extend ({\n                'key': this.apiKey,\n                'nonce': nonce,\n            }, params);\n            let request = this.urlencode (query);\n            let secret = this.hash (this.encode (this.secret));\n            query['signature'] = this.hmac (this.encode (request), this.encode (secret));\n            body = this.urlencode (query);\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('result' in response)\n            if (!response['result'])\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class btcchina extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'btcchina',\n            'name': 'BTCChina',\n            'countries': 'CN',\n            'rateLimit': 1500,\n            'version': 'v1',\n            'hasCORS': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766368-465b3286-5ed6-11e7-9a11-0f6467e1d82b.jpg',\n                'api': {\n                    'plus': 'https://plus-api.btcchina.com/market',\n                    'public': 'https://data.btcchina.com/data',\n                    'private': 'https://api.btcchina.com/api_trade_v1.php',\n                },\n                'www': 'https://www.btcchina.com',\n                'doc': 'https://www.btcchina.com/apidocs'\n            },\n            'api': {\n                'plus': {\n                    'get': [\n                        'orderbook',\n                        'ticker',\n                        'trade',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'historydata',\n                        'orderbook',\n                        'ticker',\n                        'trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'BuyIcebergOrder',\n                        'BuyOrder',\n                        'BuyOrder2',\n                        'BuyStopOrder',\n                        'CancelIcebergOrder',\n                        'CancelOrder',\n                        'CancelStopOrder',\n                        'GetAccountInfo',\n                        'getArchivedOrder',\n                        'getArchivedOrders',\n                        'GetDeposits',\n                        'GetIcebergOrder',\n                        'GetIcebergOrders',\n                        'GetMarketDepth',\n                        'GetMarketDepth2',\n                        'GetOrder',\n                        'GetOrders',\n                        'GetStopOrder',\n                        'GetStopOrders',\n                        'GetTransactions',\n                        'GetWithdrawal',\n                        'GetWithdrawals',\n                        'RequestWithdrawal',\n                        'SellIcebergOrder',\n                        'SellOrder',\n                        'SellOrder2',\n                        'SellStopOrder',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/CNY': { 'id': 'btccny', 'symbol': 'BTC/CNY', 'base': 'BTC', 'quote': 'CNY', 'api': 'public', 'plus': false },\n                'LTC/CNY': { 'id': 'ltccny', 'symbol': 'LTC/CNY', 'base': 'LTC', 'quote': 'CNY', 'api': 'public', 'plus': false },\n                'LTC/BTC': { 'id': 'ltcbtc', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC', 'api': 'public', 'plus': false },\n                'BCH/CNY': { 'id': 'bcccny', 'symbol': 'BCH/CNY', 'base': 'BCH', 'quote': 'CNY', 'api': 'plus', 'plus': true },\n                'ETH/CNY': { 'id': 'ethcny', 'symbol': 'ETH/CNY', 'base': 'ETH', 'quote': 'CNY', 'api': 'plus', 'plus': true },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetTicker ({\n            'market': 'all',\n        });\n        let result = [];\n        let keys = Object.keys (markets);\n        for (let p = 0; p < keys.length; p++) {\n            let key = keys[p];\n            let market = markets[key];\n            let parts = key.split ('_');\n            let id = parts[1];\n            let base = id.slice (0, 3);\n            let quote = id.slice (3, 6);\n            base = base.toUpperCase ();\n            quote = quote.toUpperCase ();\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetAccountInfo ();\n        let balances = response['result'];\n        let result = { 'info': balances };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let account = this.account ();\n            if (lowercase in balances['balance'])\n                account['total'] = parseFloat (balances['balance'][lowercase]['amount']);\n            if (lowercase in balances['frozen'])\n                account['used'] = parseFloat (balances['frozen'][lowercase]['amount']);\n            account['free'] = account['total'] - account['used'];\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    createMarketRequest (market) {\n        let request = {};\n        let field = (market['plus']) ? 'symbol' : 'market';\n        request[field] = market['id'];\n        return request;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = market['api'] + 'GetOrderbook';\n        let request = this.createMarketRequest (market);\n        let orderbook = await this[method] (this.extend (request, params));\n        let timestamp = orderbook['date'] * 1000;\n        let result = this.parseOrderBook (orderbook, timestamp);\n        result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    parseTicker (ticker, market) {\n        let timestamp = ticker['date'] * 1000;\n        return {\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy']),\n            'ask': parseFloat (ticker['sell']),\n            'vwap': parseFloat (ticker['vwap']),\n            'open': parseFloat (ticker['open']),\n            'close': parseFloat (ticker['prev_close']),\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['vol']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTickerPlus (ticker, market) {\n        let timestamp = ticker['Timestamp'];\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['High']),\n            'low': parseFloat (ticker['Low']),\n            'bid': parseFloat (ticker['BidPrice']),\n            'ask': parseFloat (ticker['AskPrice']),\n            'vwap': undefined,\n            'open': parseFloat (ticker['Open']),\n            'close': parseFloat (ticker['PrevCls']),\n            'first': undefined,\n            'last': parseFloat (ticker['Last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['Volume24H']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = market['api'] + 'GetTicker';\n        let request = this.createMarketRequest (market);\n        let tickers = await this[method] (this.extend (request, params));\n        let ticker = tickers['ticker'];\n        if (market['plus'])\n            return this.parseTickerPlus (ticker, market);\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        return {\n            'id': trade['tid'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    parseTradePlus (trade, market) {\n        let timestamp = this.parse8601 (trade['timestamp']);\n        return {\n            'id': undefined,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['side'].toLowerCase (),\n            'price': trade['price'],\n            'amount': trade['size'],\n        };\n    }\n\n    parseTradesPlus (trades, market = undefined) {\n        let result = [];\n        for (let i = 0; i < trades.length; i++) {\n            result.push (this.parseTradePlus (trades[i], market));\n        }\n        return result;\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = market['api'] + 'GetTrade';\n        let request = this.createMarketRequest (market);\n        if (market['plus']) {\n            let now = this.milliseconds ();\n            request['start_time'] = now - 86400 * 1000;\n            request['end_time'] = now;\n        } else {\n            method += 's'; // trades vs trade\n        }\n        let response = await this[method] (this.extend (request, params));\n        if (market['plus']) {\n            return this.parseTradesPlus (response['trades'], market);\n        }\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'privatePost' + this.capitalize (side) + 'Order2';\n        let order = {};\n        let id = market['id'].toUpperCase ();\n        if (type == 'market') {\n            order['params'] = [ undefined, amount, id ];\n        } else {\n            order['params'] = [ price, amount, id ];\n        }\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = params['market']; // TODO fixme\n        return await this.privatePostCancelOrder (this.extend ({\n            'params': [ id, market ],\n        }, params));\n    }\n\n    nonce () {\n        return this.microseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api] + '/' + path;\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            let p = [];\n            if ('params' in params)\n                p = params['params'];\n            let nonce = this.nonce ();\n            let request = {\n                'method': path,\n                'id': nonce,\n                'params': p,\n            };\n            p = p.join (',');\n            body = this.json (request);\n            let query = (\n                'tonce=' + nonce +\n                '&accesskey=' + this.apiKey +\n                '&requestmethod=' + method.toLowerCase () +\n                '&id=' + nonce +\n                '&method=' + path +\n                '&params=' + p\n            );\n            let signature = this.hmac (this.encode (query), this.encode (this.secret), 'sha1');\n            let auth = this.encode (this.apiKey + ':' + signature);\n            headers = {\n                'Authorization': 'Basic ' + this.stringToBase64 (auth),\n                'Json-Rpc-Tonce': nonce,\n            };\n        } else {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst btcturk = require ('./btcturk.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class btcexchange extends btcturk {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'btcexchange',\n            'name': 'BTCExchange',\n            'countries': 'PH', // Philippines\n            'rateLimit': 1500,\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27993052-4c92911a-64aa-11e7-96d8-ec6ac3435757.jpg',\n                'api': 'https://www.btcexchange.ph/api',\n                'www': 'https://www.btcexchange.ph',\n                'doc': 'https://github.com/BTCTrader/broker-api-docs',\n            },\n            'markets': {\n                'BTC/PHP': { 'id': 'BTC/PHP', 'symbol': 'BTC/PHP', 'base': 'BTC', 'quote': 'PHP' },\n            },\n        });\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, OrderNotFound, NotSupported } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class btcmarkets extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'btcmarkets',\n            'name': 'BTC Markets',\n            'countries': 'AU', // Australia\n            'rateLimit': 1000, // market data cached for 1 second (trades cached for 2 seconds)\n            'hasCORS': false,\n            'hasFetchOrder': true,\n            'hasFetchOrders': true,\n            'hasFetchClosedOrders': true,\n            'hasFetchOpenOrders': true,\n            'hasFetchMyTrades': true,\n            'has': {\n                'fetchOrder': true,\n                'fetchOrders': true,\n                'fetchClosedOrders': 'emulated',\n                'fetchOpenOrders': true,\n                'fetchMyTrades': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/29142911-0e1acfc2-7d5c-11e7-98c4-07d9532b29d7.jpg',\n                'api': 'https://api.btcmarkets.net',\n                'www': 'https://btcmarkets.net/',\n                'doc': 'https://github.com/BTCMarkets/API',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'market/{id}/tick',\n                        'market/{id}/orderbook',\n                        'market/{id}/trades',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'account/balance',\n                        'account/{id}/tradingfee',\n                    ],\n                    'post': [\n                        'fundtransfer/withdrawCrypto',\n                        'fundtransfer/withdrawEFT',\n                        'order/create',\n                        'order/cancel',\n                        'order/history',\n                        'order/open',\n                        'order/trade/history',\n                        'order/createBatch', // they promise it's coming soon...\n                        'order/detail',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/AUD': { 'id': 'BTC/AUD', 'symbol': 'BTC/AUD', 'base': 'BTC', 'quote': 'AUD', 'maker': 0.0085, 'taker': 0.0085 },\n                'LTC/AUD': { 'id': 'LTC/AUD', 'symbol': 'LTC/AUD', 'base': 'LTC', 'quote': 'AUD', 'maker': 0.0085, 'taker': 0.0085 },\n                'ETH/AUD': { 'id': 'ETH/AUD', 'symbol': 'ETH/AUD', 'base': 'ETH', 'quote': 'AUD', 'maker': 0.0085, 'taker': 0.0085 },\n                'ETC/AUD': { 'id': 'ETC/AUD', 'symbol': 'ETC/AUD', 'base': 'ETC', 'quote': 'AUD', 'maker': 0.0085, 'taker': 0.0085 },\n                'XRP/AUD': { 'id': 'XRP/AUD', 'symbol': 'XRP/AUD', 'base': 'XRP', 'quote': 'AUD', 'maker': 0.0085, 'taker': 0.0085 },\n                'BCH/AUD': { 'id': 'BCH/AUD', 'symbol': 'BCH/AUD', 'base': 'BCH', 'quote': 'AUD', 'maker': 0.0085, 'taker': 0.0085 },\n                'LTC/BTC': { 'id': 'LTC/BTC', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC', 'maker': 0.0022, 'taker': 0.0022 },\n                'ETH/BTC': { 'id': 'ETH/BTC', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC', 'maker': 0.0022, 'taker': 0.0022 },\n                'ETC/BTC': { 'id': 'ETC/BTC', 'symbol': 'ETC/BTC', 'base': 'ETC', 'quote': 'BTC', 'maker': 0.0022, 'taker': 0.0022 },\n                'XRP/BTC': { 'id': 'XRP/BTC', 'symbol': 'XRP/BTC', 'base': 'XRP', 'quote': 'BTC', 'maker': 0.0022, 'taker': 0.0022 },\n                'BCH/BTC': { 'id': 'BCH/BTC', 'symbol': 'BCH/BTC', 'base': 'BCH', 'quote': 'BTC', 'maker': 0.0022, 'taker': 0.0022 },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privateGetAccountBalance ();\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let multiplier = 100000000;\n            let total = parseFloat (balance['balance'] / multiplier);\n            let used = parseFloat (balance['pendingFunds'] / multiplier);\n            let free = total - used;\n            let account = {\n                'free': free,\n                'used': used,\n                'total': total,\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orderbook = await this.publicGetMarketIdOrderbook (this.extend ({\n            'id': market['id'],\n        }, params));\n        let timestamp = orderbook['timestamp'] * 1000;\n        return this.parseOrderBook (orderbook, timestamp);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['timestamp'] * 1000;\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['bestBid']),\n            'ask': parseFloat (ticker['bestAsk']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['lastPrice']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume24h']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetMarketIdTick (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['date'] * 1000;\n        return {\n            'info': trade,\n            'id': trade['tid'].toString (),\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetMarketIdTrades (this.extend ({\n            // 'since': 59868345231,\n            'id': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let multiplier = 100000000; // for price and volume\n        // does BTC Markets support market orders at all?\n        let orderSide = (side == 'buy') ? 'Bid' : 'Ask';\n        let order = this.ordered ({\n            'currency': market['quote'],\n        });\n        order['currency'] = market['quote'];\n        order['instrument'] = market['base'];\n        order['price'] = parseInt (price * multiplier);\n        order['volume'] = parseInt (amount * multiplier);\n        order['orderSide'] = orderSide;\n        order['ordertype'] = this.capitalize (type);\n        order['clientRequestId'] = this.nonce ().toString ();\n        let response = await this.privatePostOrderCreate (order);\n        return {\n            'info': response,\n            'id': response['id'].toString (),\n        };\n    }\n\n    async cancelOrders (ids) {\n        await this.loadMarkets ();\n        for (let i = 0; i < ids.length; i++) {\n            ids[i] = parseInt(ids[i]);\n        }\n        return await this.privatePostOrderCancel ({ 'orderIds': ids });\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.cancelOrders ([ id ]);\n    }\n\n    parseMyTrade (trade, market) {\n        let multiplier = 100000000;\n        let timestamp = trade['creationTime'];\n        let side = (trade['side'] == 'Bid') ? 'buy' : 'sell';\n        // BTCMarkets always charge in AUD for AUD-related transactions.\n        let currency = (market['quote'] == 'AUD') ? market['quote'] : market['base'];\n        return {\n            'info': trade,\n            'id': trade['id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': side,\n            'price': trade['price'] / multiplier,\n            'fee': {\n                'currency': currency,\n                'cost': trade['fee'] / multiplier,\n            },\n            'amount': trade['volume'] / multiplier,\n            'order': this.safeString (trade, 'orderId'),\n        };\n    }\n\n    parseMyTrades (trades, market = undefined, since = undefined, limit = undefined) {\n        let result = [];\n        for (let i = 0; i < trades.length; i++) {\n            let trade = this.parseMyTrade (trades[i], market);\n            result.push (trade);\n        }\n        return result;\n    }\n\n    parseOrder (order, market = undefined) {\n        let multiplier = 100000000;\n        let side = (order['orderSide'] == 'Bid') ? 'buy' : 'sell';\n        let type = (order['ordertype'] == 'Limit') ? 'limit' : 'market';\n        let timestamp = order['creationTime'];\n        if (!market) {\n            market = this.market(order['instrument'] + \"/\" + order['currency']);\n        }\n        let status = 'open';\n        if (order['status'] == 'Failed' || order['status'] == 'Cancelled' || order['status'] == 'Partially Cancelled' || order['status'] == 'Error') {\n            status = 'canceled';\n        } else if (order['status'] == \"Fully Matched\" || order['status'] == \"Partially Matched\") {\n            status = 'closed';\n        }\n        let price = this.safeFloat (order, 'price') / multiplier;\n        let amount = this.safeFloat (order, 'volume') / multiplier;\n        let remaining = this.safeFloat (order, 'openVolume', 0.0) / multiplier;\n        let filled = amount - remaining;\n        let cost = price * amount;\n        let trades = this.parseMyTrades (order['trades'], market);\n        let result = {\n            'info': order,\n            'id': order['id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': type,\n            'side': side,\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'trades': trades,\n            'fee': undefined,\n        };\n        return result;\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let ids = [ parseInt (id) ];\n        let response = await this.privatePostOrderDetail (this.extend ({\n            'orderIds': ids,\n        }, params));\n        let numOrders = response['orders'].length;\n        if (numOrders < 1)\n            throw new OrderNotFound (this.id + ' No matching order found: ' + id);\n        let order = response['orders'][0];\n        return this.parseOrder (order);\n    }\n\n    async prepareHistoryRequest (market, since = undefined, limit = undefined) {\n        let request = this.ordered ({\n            'currency': market['quote'],\n            'instrument': market['base'],\n        });\n        if (limit) {\n            request['limit'] = limit;\n        } else {\n            request['limit'] = 100;\n        }\n        if (since) {\n            request['since'] = since;\n        } else {\n            request['since'] = 0;\n        }\n        return request;\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new NotSupported (this.id + ': fetchOrders requires a `symbol` parameter.');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = this.prepareHistoryRequest (market, since, limit);\n        let response = await this.privatePostOrderHistory (this.extend (request, params));\n        return this.parseOrders (response['orders'], market);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new NotSupported (this.id + ': fetchOpenOrders requires a `symbol` parameter.');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = this.prepareHistoryRequest (market, since, limit);\n        let response = await this.privatePostOrderOpen (this.extend (request, params));\n        return this.parseOrders (response['orders'], market);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, since, limit, params);\n        return this.filterBy (orders, 'status', 'closed');\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new NotSupported (this.id + ': fetchMyTrades requires a `symbol` parameter.');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = this.prepareHistoryRequest (market, since, limit);\n        let response = await this.privatePostOrderTradeHistory (this.extend (request, params));\n        return this.parseMyTrades (response['trades'], market);\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let uri = '/' + this.implodeParams (path, params);\n        let url = this.urls['api'] + uri;\n        if (api == 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let auth = uri + \"\\n\" + nonce + \"\\n\";\n            headers = {\n                'Content-Type': 'application/json',\n                'apikey': this.apiKey,\n                'timestamp': nonce,\n            };\n            if (method == 'POST') {\n                body = this.json (params);\n                auth += body;\n            }\n            let secret = this.base64ToBinary (this.secret);\n            let signature = this.hmac (this.encode (auth), secret, 'sha512', 'base64');\n            headers['signature'] = this.decode (signature);\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (api == 'private') {\n            if ('success' in response)\n                if (!response['success'])\n                    throw new ExchangeError (this.id + ' ' + this.json (response));\n            return response;\n        }\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class btctradeua extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'btctradeua',\n            'name': 'BTC Trade UA',\n            'countries': 'UA', // Ukraine,\n            'rateLimit': 3000,\n            'hasCORS': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27941483-79fc7350-62d9-11e7-9f61-ac47f28fcd96.jpg',\n                'api': 'https://btc-trade.com.ua/api',\n                'www': 'https://btc-trade.com.ua',\n                'doc': 'https://docs.google.com/document/d/1ocYA0yMy_RXd561sfG3qEPZ80kyll36HUxvCRe5GbhE/edit',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'deals/{symbol}',\n                        'trades/sell/{symbol}',\n                        'trades/buy/{symbol}',\n                        'japan_stat/high/{symbol}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'auth',\n                        'ask/{symbol}',\n                        'balance',\n                        'bid/{symbol}',\n                        'buy/{symbol}',\n                        'my_orders/{symbol}',\n                        'order/status/{id}',\n                        'remove/order/{id}',\n                        'sell/{symbol}',\n                    ],\n                },\n            },\n            'markets': {\n                'BCH/UAH': { 'id': 'bch_uah', 'symbol': 'BCH/UAH', 'base': 'BCH', 'quote': 'UAH' },\n                'BTC/UAH': { 'id': 'btc_uah', 'symbol': 'BTC/UAH', 'base': 'BTC', 'quote': 'UAH', 'precision': { 'price': 1 }, 'limits': { 'amount': { 'min': 0.0000000001 }}},\n                'DASH/BTC': { 'id': 'dash_btc', 'symbol': 'DASH/BTC', 'base': 'DASH', 'quote': 'BTC' },\n                'DASH/UAH': { 'id': 'dash_uah', 'symbol': 'DASH/UAH', 'base': 'DASH', 'quote': 'UAH' },\n                'DOGE/BTC': { 'id': 'doge_btc', 'symbol': 'DOGE/BTC', 'base': 'DOGE', 'quote': 'BTC' },\n                'DOGE/UAH': { 'id': 'doge_uah', 'symbol': 'DOGE/UAH', 'base': 'DOGE', 'quote': 'UAH' },\n                'ETH/UAH': { 'id': 'eth_uah', 'symbol': 'ETH/UAH', 'base': 'ETH', 'quote': 'UAH' },\n                'ITI/UAH': { 'id': 'iti_uah', 'symbol': 'ITI/UAH', 'base': 'ITI', 'quote': 'UAH' },\n                'KRB/UAH': { 'id': 'krb_uah', 'symbol': 'KRB/UAH', 'base': 'KRB', 'quote': 'UAH' },\n                'LTC/BTC': { 'id': 'ltc_btc', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC' },\n                'LTC/UAH': { 'id': 'ltc_uah', 'symbol': 'LTC/UAH', 'base': 'LTC', 'quote': 'UAH' },\n                'NVC/BTC': { 'id': 'nvc_btc', 'symbol': 'NVC/BTC', 'base': 'NVC', 'quote': 'BTC' },\n                'NVC/UAH': { 'id': 'nvc_uah', 'symbol': 'NVC/UAH', 'base': 'NVC', 'quote': 'UAH' },\n                'PPC/BTC': { 'id': 'ppc_btc', 'symbol': 'PPC/BTC', 'base': 'PPC', 'quote': 'BTC' },\n                'SIB/UAH': { 'id': 'sib_uah', 'symbol': 'SIB/UAH', 'base': 'SIB', 'quote': 'UAH' },\n                'XMR/UAH': { 'id': 'xmr_uah', 'symbol': 'XMR/UAH', 'base': 'XMR', 'quote': 'UAH' },\n                'ZEC/UAH': { 'id': 'zec_uah', 'symbol': 'ZEC/UAH', 'base': 'ZEC', 'quote': 'UAH' },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.1 / 100,\n                    'taker': 0.1 / 100,\n                },\n                'funding': {\n                    'withdraw': {\n                        'BTC': 0.0006,\n                        'LTC': 0.01,\n                        'NVC': 0.01,\n                        'DOGE': 10,\n                    },\n                },\n            },\n        });\n    }\n\n    signIn () {\n        return this.privatePostAuth ();\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostBalance ();\n        let result = { 'info': response };\n        if ('accounts' in response) {\n            let accounts = response['accounts'];\n            for (let b = 0; b < accounts.length; b++) {\n                let account = accounts[b];\n                let currency = account['currency'];\n                let balance = parseFloat (account['balance']);\n                result[currency] = {\n                    'free': balance,\n                    'used': 0.0,\n                    'total': balance,\n                };\n            }\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let market = this.market (symbol);\n        let bids = await this.publicGetTradesBuySymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        let asks = await this.publicGetTradesSellSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        let orderbook = {\n            'bids': [],\n            'asks': [],\n        };\n        if (bids) {\n            if ('list' in bids)\n                orderbook['bids'] = bids['list'];\n        }\n        if (asks) {\n            if ('list' in asks)\n                orderbook['asks'] = asks['list'];\n        }\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'currency_trade');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let response = await this.publicGetJapanStatHighSymbol (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        let orderbook = await this.fetchOrderBook (symbol);\n        let bid = undefined;\n        let numBids = orderbook['bids'].length;\n        if (numBids > 0)\n            bid = orderbook['bids'][0][0];\n        let ask = undefined;\n        let numAsks = orderbook['asks'].length;\n        if (numAsks > 0)\n            ask = orderbook['asks'][0][0];\n        let ticker = response['trades'];\n        let timestamp = this.milliseconds ();\n        let result = {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': bid,\n            'ask': ask,\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': undefined,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n        let tickerLength = ticker.length;\n        if (tickerLength > 0) {\n            let start = Math.max (tickerLength - 48, 0);\n            for (let t = start; t < ticker.length; t++) {\n                let candle = ticker[t];\n                if (typeof result['open'] == 'undefined')\n                    result['open'] = candle[1];\n                if ((typeof result['high'] == 'undefined') || (result['high'] < candle[2]))\n                    result['high'] = candle[2];\n                if ((typeof result['low'] == 'undefined') || (result['low'] > candle[3]))\n                    result['low'] = candle[3];\n                if (typeof result['baseVolume'] == 'undefined')\n                    result['baseVolume'] = -candle[5];\n                else\n                    result['baseVolume'] -= candle[5];\n            }\n            let last = tickerLength - 1;\n            result['close'] = ticker[last][4];\n            result['baseVolume'] = -1 * result['baseVolume'];\n        }\n        return result;\n    }\n\n    convertCyrillicMonthNameToString (cyrillic) {\n        let months = {\n            'января': '01',\n            'февраля': '02',\n            'марта': '03',\n            'апреля': '04',\n            'мая': '05',\n            'июня': '06',\n            'июля': '07',\n            'августа': '08',\n            'сентября': '09',\n            'октября': '10',\n            'ноября': '11',\n            'декабря': '12',\n        };\n        let month = undefined;\n        if (cyrillic in months)\n            month = months[cyrillic];\n        return month;\n    }\n\n    parseCyrillicDatetime (cyrillic) {\n        let parts = cyrillic.split (' ');\n        let day = parts[0];\n        let month = this.convertCyrillicMonthNameToString (parts[1]);\n        if (!month)\n            throw new ExchangeError (this.id + ' parseTrade() undefined month name: ' + cyrillic);\n        let year = parts[2];\n        let hms = parts[4];\n        let hmsLength = hms.length;\n        if (hmsLength == 7) {\n            hms = '0' + hms;\n        }\n        if (day.length == 1) {\n            day = '0' + day;\n        }\n        let ymd = [ year, month, day ].join ('-');\n        let ymdhms = ymd + 'T' + hms;\n        let timestamp = this.parse8601 (ymdhms);\n        // server reports local time, adjust to UTC\n        let md = [ month, day ].join ('');\n        md = parseInt (md);\n        // a special case for DST\n        // subtract 2 hours during winter\n        if (md < 325 || md > 1028)\n            return timestamp - 7200000;\n        // subtract 3 hours during summer\n        return timestamp - 10800000;\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = this.parseCyrillicDatetime (trade['pub_date']);\n        return {\n            'id': trade['id'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': 'limit',\n            'side': trade['type'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amnt_trade']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetDealsSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        // they report each trade twice (once for both of the two sides of the fill)\n        // deduplicate trades for that reason\n        let trades = [];\n        for (let i = 0; i < response.length; i++) {\n            if (response[i]['id'] % 2) {\n                trades.push (response[i]);\n            }\n        }\n        return this.parseTrades (trades, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type == 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let market = this.market (symbol);\n        let method = 'privatePost' + this.capitalize (side) + 'Id';\n        let order = {\n            'count': amount,\n            'currency1': market['quote'],\n            'currency': market['base'],\n            'price': price,\n        };\n        return this[method] (this.extend (order, params));\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostRemoveOrderId ({ 'id': id });\n    }\n\n    parseOrder (trade, market) {\n        let timestamp = this.milliseconds;\n        return {\n            'id': trade['id'],\n            'timestamp': timestamp, // until they fix their timestamp\n            'datetime': this.iso8601 (timestamp),\n            'status': 'open',\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['type'],\n            'price': trade['price'],\n            'amount': trade['amnt_trade'],\n            'filled': 0,\n            'remaining': trade['amnt_trade'],\n            'trades': undefined,\n            'info': trade,\n        };\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOpenOrders requires a symbol param');\n        let market = this.market (symbol);\n        let response = await this.privatePostMyOrdersSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        let orders = response['your_open_orders'];\n        return this.parseOrders (orders, market, since, limit);\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += this.implodeParams (path, query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({\n                'out_order_id': nonce,\n                'nonce': nonce,\n            }, query));\n            let auth = body + this.secret;\n            headers = {\n                'public-key': this.apiKey,\n                'api-sign': this.hash (this.encode (auth), 'sha256'),\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class btcturk extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'btcturk',\n            'name': 'BTCTurk',\n            'countries': 'TR', // Turkey\n            'rateLimit': 1000,\n            'hasCORS': true,\n            'hasFetchTickers': true,\n            'hasFetchOHLCV': true,\n            'timeframes': {\n                '1d': '1d',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27992709-18e15646-64a3-11e7-9fa2-b0950ec7712f.jpg',\n                'api': 'https://www.btcturk.com/api',\n                'www': 'https://www.btcturk.com',\n                'doc': 'https://github.com/BTCTrader/broker-api-docs',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'ohlcdata', // ?last=COUNT\n                        'orderbook',\n                        'ticker',\n                        'trades',   // ?last=COUNT (max 50)\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'balance',\n                        'openOrders',\n                        'userTransactions', // ?offset=0&limit=25&sort=asc\n                    ],\n                    'post': [\n                        'buy',\n                        'cancelOrder',\n                        'sell',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/TRY': { 'id': 'BTCTRY', 'symbol': 'BTC/TRY', 'base': 'BTC', 'quote': 'TRY', 'maker': 0.002 * 1.18, 'taker': 0.0035 * 1.18 },\n                'ETH/TRY': { 'id': 'ETHTRY', 'symbol': 'ETH/TRY', 'base': 'ETH', 'quote': 'TRY', 'maker': 0.002 * 1.18, 'taker': 0.0035 * 1.18 },\n                'ETH/BTC': { 'id': 'ETHBTC', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC', 'maker': 0.002 * 1.18, 'taker': 0.0035 * 1.18 },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privateGetBalance ();\n        let result = { 'info': response };\n        let base = {\n            'free': response['bitcoin_available'],\n            'used': response['bitcoin_reserved'],\n            'total': response['bitcoin_balance'],\n        };\n        let quote = {\n            'free': response['money_available'],\n            'used': response['money_reserved'],\n            'total': response['money_balance'],\n        };\n        let symbol = this.symbols[0];\n        let market = this.markets[symbol];\n        result[market['base']] = base;\n        result[market['quote']] = quote;\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let market = this.market (symbol);\n        let orderbook = await this.publicGetOrderbook (this.extend ({\n            'pairSymbol': market['id'],\n        }, params));\n        let timestamp = parseInt (orderbook['timestamp'] * 1000);\n        return this.parseOrderBook (orderbook, timestamp);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = parseInt (ticker['timestamp']) * 1000;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': parseFloat (ticker['open']),\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': parseFloat (ticker['average']),\n            'baseVolume': parseFloat (ticker['volume']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTicker (params);\n        let result = {};\n        for (let i = 0; i < tickers.length; i++) {\n            let ticker = tickers[i];\n            let symbol = ticker['pair'];\n            let market = undefined;\n            if (symbol in this.markets_by_id) {\n                market = this.markets_by_id[symbol];\n                symbol = market['symbol'];\n            }\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.fetchTickers ();\n        let result = undefined;\n        if (symbol in tickers)\n            result = tickers[symbol];\n        return result;\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['date'] * 1000;\n        return {\n            'id': trade['tid'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        // let maxCount = 50;\n        let response = await this.publicGetTrades (this.extend ({\n            'pairSymbol': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1d', since = undefined, limit = undefined) {\n        let timestamp = this.parse8601 (ohlcv['Time']);\n        return [\n            timestamp,\n            ohlcv['Open'],\n            ohlcv['High'],\n            ohlcv['Low'],\n            ohlcv['Close'],\n            ohlcv['Volume'],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1d', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {};\n        if (limit)\n            request['last'] = limit;\n        let response = await this.publicGetOhlcdata (this.extend (request, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let method = 'privatePost' + this.capitalize (side);\n        let order = {\n            'Type': (side == 'buy') ? 'BuyBtc' : 'SelBtc',\n            'IsMarketOrder': (type == 'market') ? 1 : 0,\n        };\n        if (type == 'market') {\n            if (side == 'buy')\n                order['Total'] = amount;\n            else\n                order['Amount'] = amount;\n        } else {\n            order['Price'] = price;\n            order['Amount'] = amount;\n        }\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelOrder ({ 'id': id });\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        if (this.id == 'btctrader')\n            throw new ExchangeError (this.id + ' is an abstract base API for BTCExchange, BTCTurk');\n        let url = this.urls['api'] + '/' + path;\n        if (api == 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            body = this.urlencode (params);\n            let secret = this.base64ToBinary (this.secret);\n            let auth = this.apiKey + nonce;\n            headers = {\n                'X-PCK': this.apiKey,\n                'X-Stamp': nonce,\n                'X-Signature': this.stringToBase64(this.hmac (this.encode (auth), secret, 'sha256', 'binary')),\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class btcx extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'btcx',\n            'name': 'BTCX',\n            'countries': [ 'IS', 'US', 'EU' ],\n            'rateLimit': 1500, // support in english is very poor, unable to tell rate limits\n            'version': 'v1',\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766385-9fdcc98c-5ed6-11e7-8f14-66d5e5cd47e6.jpg',\n                'api': 'https://btc-x.is/api',\n                'www': 'https://btc-x.is',\n                'doc': 'https://btc-x.is/custom/api-document.html',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'depth/{id}/{limit}',\n                        'ticker/{id}',\n                        'trade/{id}/{limit}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balance',\n                        'cancel',\n                        'history',\n                        'order',\n                        'redeem',\n                        'trade',\n                        'withdraw',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/USD': { 'id': 'btc/usd', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD' },\n                'BTC/EUR': { 'id': 'btc/eur', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR' },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let balances = await this.privatePostBalance ();\n        let result = { 'info': balances };\n        let currencies = Object.keys (balances);\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let uppercase = currency.toUpperCase ();\n            let account = {\n                'free': balances[currency],\n                'used': 0.0,\n                'total': balances[currency],\n            };\n            result[uppercase] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetDepthIdLimit (this.extend ({\n            'id': this.marketId (symbol),\n            'limit': 1000,\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'amount');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetTickerId (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        let timestamp = ticker['time'] * 1000;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['sell']),\n            'ask': parseFloat (ticker['buy']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': parseFloat (ticker['volume']),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        let side = (trade['type'] == 'ask') ? 'sell' : 'buy';\n        return {\n            'id': trade['id'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': side,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetTradeIdLimit (this.extend ({\n            'id': market['id'],\n            'limit': 1000,\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let response = await this.privatePostTrade (this.extend ({\n            'type': side.toUpperCase (),\n            'market': this.marketId (symbol),\n            'amount': amount,\n            'price': price,\n        }, params));\n        return {\n            'info': response,\n            'id': response['order']['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancel ({ 'order': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/';\n        if (api == 'public') {\n            url += this.implodeParams (path, params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            url += api;\n            body = this.urlencode (this.extend ({\n                'Method': path.toUpperCase (),\n                'Nonce': nonce,\n            }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Key': this.apiKey,\n                'Signature': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bter extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bter',\n            'name': 'Bter',\n            'countries': [ 'VG', 'CN' ], // British Virgin Islands, China\n            'version': '2',\n            // obsolete metainfo interface\n            'hasCORS': false,\n            'hasFetchTickers': true,\n            'hasWithdraw': true,\n            // new metainfo interface\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27980479-cfa3188c-6387-11e7-8191-93fc4184ba5c.jpg',\n                'api': {\n                    'public': 'https://data.bter.com/api',\n                    'private': 'https://api.bter.com/api',\n                },\n                'www': 'https://bter.com',\n                'doc': 'https://bter.com/api2',\n                'fees': 'https://bter.com/fee',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'pairs',\n                        'marketinfo',\n                        'marketlist',\n                        'tickers',\n                        'ticker/{id}',\n                        'orderBook/{id}',\n                        'trade/{id}',\n                        'tradeHistory/{id}',\n                        'tradeHistory/{id}/{tid}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balances',\n                        'depositAddress',\n                        'newAddress',\n                        'depositsWithdrawals',\n                        'buy',\n                        'sell',\n                        'cancelOrder',\n                        'cancelAllOrders',\n                        'getOrder',\n                        'openOrders',\n                        'tradeHistory',\n                        'withdraw',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicGetMarketinfo ();\n        let markets = this.safeValue (response, 'pairs');\n        if (!markets)\n            throw new ExchangeError (this.id + ' fetchMarkets got an unrecognized response');\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let keys = Object.keys (market);\n            let id = keys[0];\n            let details = market[id];\n            let [ base, quote ] = id.split ('_');\n            base = base.toUpperCase ();\n            quote = quote.toUpperCase ();\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            let precision = {\n                'amount': details['decimal_places'],\n                'price': details['decimal_places'],\n            };\n            let amountLimits = {\n                'min': details['min_amount'],\n                'max': undefined,\n            };\n            let priceLimits = {\n                'min': undefined,\n                'max': undefined,\n            };\n            let limits = {\n                'amount': amountLimits,\n                'price': priceLimits,\n            };\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n                'maker': details['fee'] / 100,\n                'taker': details['fee'] / 100,\n                'precision': precision,\n                'limits': limits,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balance = await this.privatePostBalances ();\n        let result = { 'info': balance };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let code = this.commonCurrencyCode (currency);\n            let account = this.account ();\n            if ('available' in balance) {\n                if (currency in balance['available']) {\n                    account['free'] = parseFloat (balance['available'][currency]);\n                }\n            }\n            if ('locked' in balance) {\n                if (currency in balance['locked']) {\n                    account['used'] = parseFloat (balance['locked'][currency]);\n                }\n            }\n            account['total'] = this.sum (account['free'], account['used']);\n            result[code] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderBookId (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        let result = this.parseOrderBook (orderbook);\n        result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high24hr']),\n            'low': parseFloat (ticker['low24hr']),\n            'bid': parseFloat (ticker['highestBid']),\n            'ask': parseFloat (ticker['lowestAsk']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': parseFloat (ticker['percentChange']),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['quoteVolume']),\n            'quoteVolume': parseFloat (ticker['baseVolume']),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTickers (params);\n        let result = {};\n        let ids = Object.keys (tickers);\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let [ baseId, quoteId ] = id.split ('_');\n            let base = baseId.toUpperCase ();\n            let quote = quoteId.toUpperCase ();\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            let ticker = tickers[id];\n            let market = undefined;\n            if (symbol in this.markets)\n                market = this.markets[symbol];\n            if (id in this.markets_by_id)\n                market = this.markets_by_id[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetTickerId (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = this.parse8601 (trade['date']);\n        return {\n            'id': trade['tradeID'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['type'],\n            'price': trade['rate'],\n            'amount': this.safeFloat (trade, 'amount'),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTradeHistoryId (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTrades (response['data'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type === 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        await this.loadMarkets ();\n        let method = 'privatePost' + this.capitalize (side);\n        let order = {\n            'currencyPair': this.marketId (symbol),\n            'rate': price,\n            'amount': amount,\n        };\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['orderNumber'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelOrder ({ 'orderNumber': id });\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostWithdraw (this.extend ({\n            'currency': currency.toLowerCase (),\n            'amount': amount,\n            'address': address, // Address must exist in you AddressBook in security settings\n        }, params));\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let prefix = (api === 'private') ? (api + '/') : '';\n        let url = this.urls['api'][api] + this.version + '/1/' + prefix + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let request = { 'nonce': nonce };\n            body = this.urlencode (this.extend (request, query));\n            let signature = this.hmac (this.encode (body), this.encode (this.secret), 'sha512');\n            headers = {\n                'Key': this.apiKey,\n                'Sign': signature,\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('result' in response)\n            if (response['result'] !== 'true')\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bxinth extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bxinth',\n            'name': 'BX.in.th',\n            'countries': 'TH', // Thailand\n            'rateLimit': 1500,\n            'hasCORS': false,\n            'hasFetchTickers': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766412-567b1eb4-5ed7-11e7-94a8-ff6a3884f6c5.jpg',\n                'api': 'https://bx.in.th/api',\n                'www': 'https://bx.in.th',\n                'doc': 'https://bx.in.th/info/api',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        '', // ticker\n                        'options',\n                        'optionbook',\n                        'orderbook',\n                        'pairing',\n                        'trade',\n                        'tradehistory',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balance',\n                        'biller',\n                        'billgroup',\n                        'billpay',\n                        'cancel',\n                        'deposit',\n                        'getorders',\n                        'history',\n                        'option-issue',\n                        'option-bid',\n                        'option-sell',\n                        'option-myissue',\n                        'option-mybid',\n                        'option-myoptions',\n                        'option-exercise',\n                        'option-cancel',\n                        'option-history',\n                        'order',\n                        'withdrawal',\n                        'withdrawal-history',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'taker': 0.25 / 100,\n                    'maker': 0.25 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetPairing ();\n        let keys = Object.keys (markets);\n        let result = [];\n        for (let p = 0; p < keys.length; p++) {\n            let market = markets[keys[p]];\n            let id = market['pairing_id'].toString ();\n            let base = market['secondary_currency'];\n            let quote = market['primary_currency'];\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    commonCurrencyCode (currency) {\n        // why would they use three letters instead of four for currency codes\n        if (currency == 'DAS')\n            return 'DASH';\n        if (currency == 'DOG')\n            return 'DOGE';\n        return currency;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostBalance ();\n        let balance = response['balance'];\n        let result = { 'info': balance };\n        let currencies = Object.keys (balance);\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let code = this.commonCurrencyCode (currency);\n            let account = {\n                'free': parseFloat (balance[currency]['available']),\n                'used': 0.0,\n                'total': parseFloat (balance[currency]['total']),\n            };\n            account['used'] = account['total'] - account['free'];\n            result[code] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderbook (this.extend ({\n            'pairing': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['orderbook']['bids']['highbid']),\n            'ask': parseFloat (ticker['orderbook']['asks']['highbid']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last_price']),\n            'change': parseFloat (ticker['change']),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume_24hours']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGet (params);\n        let result = {};\n        let ids = Object.keys (tickers);\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let ticker = tickers[id];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let tickers = await this.publicGet (this.extend ({\n            'pairing': market['id'],\n        }, params));\n        let id = market['id'].toString ();\n        let ticker = tickers[id];\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = this.parse8601 (trade['trade_date']);\n        return {\n            'id': trade['trade_id'],\n            'info': trade,\n            'order': trade['order_id'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['trade_type'],\n            'price': parseFloat (trade['rate']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTrade (this.extend ({\n            'pairing': market['id'],\n        }, params));\n        return this.parseTrades (response['trades'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostOrder (this.extend ({\n            'pairing': this.marketId (symbol),\n            'type': side,\n            'amount': amount,\n            'rate': price,\n        }, params));\n        return {\n            'info': response,\n            'id': response['order_id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let pairing = undefined; // TODO fixme\n        return await this.privatePostCancel ({\n            'order_id': id,\n            'pairing': pairing,\n        });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/';\n        if (path)\n            url += path + '/';\n        if (Object.keys (params).length)\n            url += '?' + this.urlencode (params);\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let auth = this.apiKey + nonce.toString () + this.secret;\n            let signature = this.hash (this.encode (auth), 'sha256');\n            body = this.urlencode (this.extend ({\n                'key': this.apiKey,\n                'nonce': nonce,\n                'signature': signature,\n                // twofa: this.twofa,\n            }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (api == 'public')\n            return response;\n        if ('success' in response)\n            if (response['success'])\n                return response;\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class ccex extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'ccex',\n            'name': 'C-CEX',\n            'countries': [ 'DE', 'EU' ],\n            'rateLimit': 1500,\n            'hasCORS': false,\n            'hasFetchTickers': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766433-16881f90-5ed8-11e7-92f8-3d92cc747a6c.jpg',\n                'api': {\n                    'tickers': 'https://c-cex.com/t',\n                    'public': 'https://c-cex.com/t/api_pub.html',\n                    'private': 'https://c-cex.com/t/api.html',\n                },\n                'www': 'https://c-cex.com',\n                'doc': 'https://c-cex.com/?id=api',\n            },\n            'api': {\n                'tickers': {\n                    'get': [\n                        'coinnames',\n                        '{market}',\n                        'pairs',\n                        'prices',\n                        'volume_{coin}',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'balancedistribution',\n                        'markethistory',\n                        'markets',\n                        'marketsummaries',\n                        'orderbook',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'buylimit',\n                        'cancel',\n                        'getbalance',\n                        'getbalances',\n                        'getopenorders',\n                        'getorder',\n                        'getorderhistory',\n                        'mytrades',\n                        'selllimit',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'taker': 0.2 / 100,\n                    'maker': 0.2 / 100,\n                },\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        if (currency == 'IOT')\n            return 'IoTcoin';\n        if (currency == 'BLC')\n            return 'Cryptobullcoin';\n        if (currency == 'XID')\n            return 'InternationalDiamond';\n        return currency;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetMarkets ();\n        let result = [];\n        for (let p = 0; p < markets['result'].length; p++) {\n            let market = markets['result'][p];\n            let id = market['MarketName'];\n            let base = market['MarketCurrency'];\n            let quote = market['BaseCurrency'];\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            }));\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetBalances ();\n        let balances = response['result'];\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let code = balance['Currency'];\n            let currency = this.commonCurrencyCode (code);\n            let account = {\n                'free': balance['Available'],\n                'used': balance['Pending'],\n                'total': balance['Balance'],\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetOrderbook (this.extend ({\n            'market': this.marketId (symbol),\n            'type': 'both',\n            'depth': 100,\n        }, params));\n        let orderbook = response['result'];\n        return this.parseOrderBook (orderbook, undefined, 'buy', 'sell', 'Rate', 'Quantity');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['updated'] * 1000;\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy']),\n            'ask': parseFloat (ticker['sell']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['lastprice']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': parseFloat (ticker['avg']),\n            'baseVolume': undefined,\n            'quoteVolume': this.safeFloat (ticker, 'buysupport'),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.tickersGetPrices (params);\n        let result = { 'info': tickers };\n        let ids = Object.keys (tickers);\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let ticker = tickers[id];\n            let uppercase = id.toUpperCase ();\n            let market = undefined;\n            let symbol = undefined;\n            if (uppercase in this.markets_by_id) {\n                market = this.markets_by_id[uppercase];\n                symbol = market['symbol'];\n            } else {\n                let [ base, quote ] = uppercase.split ('-');\n                base = this.commonCurrencyCode (base);\n                quote = this.commonCurrencyCode (quote);\n                symbol = base + '/' + quote;\n            }\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.tickersGetMarket (this.extend ({\n            'market': market['id'].toLowerCase (),\n        }, params));\n        let ticker = response['ticker'];\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = this.parse8601 (trade['TimeStamp']);\n        return {\n            'id': trade['Id'],\n            'info': trade,\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['OrderType'].toLowerCase (),\n            'price': trade['Price'],\n            'amount': trade['Quantity'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetMarkethistory (this.extend ({\n            'market': market['id'],\n            'type': 'both',\n            'depth': 100,\n        }, params));\n        return this.parseTrades (response['result'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let method = 'privateGet' + this.capitalize (side) + type;\n        let response = await this[method] (this.extend ({\n            'market': this.marketId (symbol),\n            'quantity': amount,\n            'rate': price,\n        }, params));\n        return {\n            'info': response,\n            'id': response['result']['uuid'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privateGetCancel ({ 'uuid': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let query = this.keysort (this.extend ({\n                'a': path,\n                'apikey': this.apiKey,\n                'nonce': nonce,\n            }, params));\n            url += '?' + this.urlencode (query);\n            headers = { 'apisign': this.hmac (this.encode (url), this.encode (this.secret), 'sha512') };\n        } else if (api == 'public') {\n            url += '?' + this.urlencode (this.extend ({\n                'a': 'get' + path,\n            }, params));\n        } else {\n            url += '/' + this.implodeParams (path, params) + '.json';\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (api == 'tickers')\n            return response;\n        if ('success' in response)\n            if (response['success'])\n                return response;\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InvalidOrder } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class cex extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'cex',\n            'name': 'CEX.IO',\n            'countries': [ 'GB', 'EU', 'CY', 'RU' ],\n            'rateLimit': 1500,\n            'hasCORS': true,\n            'hasFetchTickers': true,\n            'hasFetchOHLCV': true,\n            'hasFetchOpenOrders': true,\n            'timeframes': {\n                '1m': '1m',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766442-8ddc33b0-5ed8-11e7-8b98-f786aef0f3c9.jpg',\n                'api': 'https://cex.io/api',\n                'www': 'https://cex.io',\n                'doc': 'https://cex.io/cex-api',\n                'fees': [\n                    'https://cex.io/fee-schedule',\n                    'https://cex.io/limits-commissions',\n                ],\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'uid': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'currency_limits/',\n                        'last_price/{pair}/',\n                        'last_prices/{currencies}/',\n                        'ohlcv/hd/{yyyymmdd}/{pair}',\n                        'order_book/{pair}/',\n                        'ticker/{pair}/',\n                        'tickers/{currencies}/',\n                        'trade_history/{pair}/',\n                    ],\n                    'post': [\n                        'convert/{pair}',\n                        'price_stats/{pair}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'active_orders_status/',\n                        'archived_orders/{pair}/',\n                        'balance/',\n                        'cancel_order/',\n                        'cancel_orders/{pair}/',\n                        'cancel_replace_order/{pair}/',\n                        'close_position/{pair}/',\n                        'get_address/',\n                        'get_myfee/',\n                        'get_order/',\n                        'get_order_tx/',\n                        'open_orders/{pair}/',\n                        'open_orders/',\n                        'open_position/{pair}/',\n                        'open_positions/{pair}/',\n                        'place_order/{pair}/',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.16 / 100,\n                    'taker': 0.25 / 100,\n                },\n                'funding': {\n                    'withdraw': {\n                        // 'USD': undefined,\n                        // 'EUR': undefined,\n                        // 'RUB': undefined,\n                        // 'GBP': undefined,\n                        'BTC': 0.001,\n                        'ETH': 0.01,\n                        'BCH': 0.001,\n                        'DASH': 0.01,\n                        'BTG': 0.001,\n                        'ZEC': 0.001,\n                        'XRP': 0.02,\n                        'XLM': undefined,\n                    },\n                    'deposit': {\n                        // 'USD': amount => amount * 0.035 + 0.25,\n                        // 'EUR': amount => amount * 0.035 + 0.24,\n                        // 'RUB': amount => amount * 0.05 + 15.57,\n                        // 'GBP': amount => amount * 0.035 + 0.2,\n                        'BTC': 0.0,\n                        'ETH': 0.0,\n                        'BCH': 0.0,\n                        'DASH': 0.0,\n                        'BTG': 0.0,\n                        'ZEC': 0.0,\n                        'XRP': 0.0,\n                        'XLM': 0.0,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetCurrencyLimits ();\n        let result = [];\n        for (let p = 0; p < markets['data']['pairs'].length; p++) {\n            let market = markets['data']['pairs'][p];\n            let id = market['symbol1'] + '/' + market['symbol2'];\n            let symbol = id;\n            let [ base, quote ] = symbol.split ('/');\n            result.push ({\n                'id': id,\n                'info': market,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'precision': {\n                    'price': this.precisionFromString (market['minPrice']),\n                    'amount': -1 * Math.log10 (market['minLotSize']),\n                },\n                'limits': {\n                    'amount': {\n                        'min': market['minLotSize'],\n                        'max': market['maxLotSize'],\n                    },\n                    'price': {\n                        'min': parseFloat (market['minPrice']),\n                        'max': parseFloat (market['maxPrice']),\n                    },\n                    'cost': {\n                        'min': market['minLotSizeS2'],\n                        'max': undefined,\n                    },\n                },\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostBalance ();\n        let result = { 'info': response };\n        let ommited = [ 'username', 'timestamp' ];\n        let balances = this.omit (response, ommited);\n        let currencies = Object.keys (balances);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            if (currency in balances) {\n                let account = {\n                    'free': this.safeFloat (balances[currency], 'available', 0.0),\n                    'used': this.safeFloat (balances[currency], 'orders', 0.0),\n                    'total': 0.0,\n                };\n                account['total'] = this.sum (account['free'], account['used']);\n                result[currency] = account;\n            }\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderBookPair (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        let timestamp = orderbook['timestamp'] * 1000;\n        return this.parseOrderBook (orderbook, timestamp);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv[0] * 1000,\n            ohlcv[1],\n            ohlcv[2],\n            ohlcv[3],\n            ohlcv[4],\n            ohlcv[5],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        if (!since)\n            since = this.milliseconds () - 86400000; // yesterday\n        let ymd = this.Ymd (since);\n        ymd = ymd.split ('-');\n        ymd = ymd.join ('');\n        let request = {\n            'pair': market['id'],\n            'yyyymmdd': ymd,\n        };\n        let response = await this.publicGetOhlcvHdYyyymmddPair (this.extend (request, params));\n        let key = 'data' + this.timeframes[timeframe];\n        let ohlcvs = JSON.parse (response[key]);\n        return this.parseOHLCVs (ohlcvs, market, timeframe, since, limit);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = undefined;\n        let iso8601 = undefined;\n        if ('timestamp' in ticker) {\n            timestamp = parseInt (ticker['timestamp']) * 1000;\n            iso8601 = this.iso8601 (timestamp);\n        }\n        let volume = this.safeFloat (ticker, 'volume');\n        let high = this.safeFloat (ticker, 'high');\n        let low = this.safeFloat (ticker, 'low');\n        let bid = this.safeFloat (ticker, 'bid');\n        let ask = this.safeFloat (ticker, 'ask');\n        let last = this.safeFloat (ticker, 'last');\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': iso8601,\n            'high': high,\n            'low': low,\n            'bid': bid,\n            'ask': ask,\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': last,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': volume,\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let currencies = Object.keys (this.currencies);\n        let response = await this.publicGetTickersCurrencies (this.extend ({\n            'currencies': currencies.join ('/'),\n        }, params));\n        let tickers = response['data'];\n        let result = {};\n        for (let t = 0; t < tickers.length; t++) {\n            let ticker = tickers[t];\n            let symbol = ticker['pair'].replace (':', '/');\n            let market = this.markets[symbol];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetTickerPair (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        return {\n            'info': trade,\n            'id': trade['tid'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['type'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTradeHistoryPair (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'pair': this.marketId (symbol),\n            'type': side,\n            'amount': amount,\n        };\n        if (type === 'limit') {\n            order['price'] = price;\n        } else {\n            // for market buy CEX.io requires the amount of quote currency to spend\n            if (side === 'buy') {\n                if (!price) {\n                    throw new InvalidOrder ('For market buy orders ' + this.id + \" requires the amount of quote currency to spend, to calculate proper costs call createOrder (symbol, 'market', 'buy', amount, price)\");\n                }\n                order['amount'] = amount * price;\n            }\n            order['order_type'] = type;\n        }\n        let response = await this.privatePostPlaceOrderPair (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelOrder ({ 'id': id });\n    }\n\n    parseOrder (order, market = undefined) {\n        let timestamp = parseInt (order['time']);\n        let symbol = undefined;\n        if (!market) {\n            let symbol = order['symbol1'] + '/' + order['symbol2'];\n            if (symbol in this.markets)\n                market = this.market (symbol);\n        }\n        let status = order['status'];\n        if (status === 'a') {\n            status = 'open'; // the unified status\n        } else if (status === 'cd') {\n            status = 'canceled';\n        } else if (status === 'c') {\n            status = 'canceled';\n        } else if (status === 'd') {\n            status = 'closed';\n        }\n        let price = this.safeFloat (order, 'price');\n        let amount = this.safeFloat (order, 'amount');\n        let remaining = this.safeFloat (order, 'pending');\n        if (!remaining)\n            remaining = this.safeFloat (order, 'remains');\n        let filled = amount - remaining;\n        let fee = undefined;\n        let cost = undefined;\n        if (market) {\n            symbol = market['symbol'];\n            cost = this.safeFloat (order, 'ta:' + market['quote']);\n            let baseFee = 'fa:' + market['base'];\n            let quoteFee = 'fa:' + market['quote'];\n            let feeRate = this.safeFloat (order, 'tradingFeeMaker');\n            if (!feeRate)\n                feeRate = this.safeFloat (order, 'tradingFeeTaker', feeRate);\n            if (feeRate)\n                feeRate /= 100.0; // convert to mathematically-correct percentage coefficients: 1.0 = 100%\n            if (baseFee in order) {\n                fee = {\n                    'currency': market['base'],\n                    'rate': feeRate,\n                    'cost': this.safeFloat (order, baseFee),\n                };\n            } else if (quoteFee in order) {\n                fee = {\n                    'currency': market['quote'],\n                    'rate': feeRate,\n                    'cost': this.safeFloat (order, quoteFee),\n                };\n            }\n        }\n        if (!cost)\n            cost = price * filled;\n        return {\n            'id': order['id'],\n            'datetime': this.iso8601 (timestamp),\n            'timestamp': timestamp,\n            'status': status,\n            'symbol': symbol,\n            'type': undefined,\n            'side': order['type'],\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'trades': undefined,\n            'fee': fee,\n            'info': order,\n        };\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        let method = 'privatePostOpenOrders';\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['pair'] = market['id'];\n            method += 'Pair';\n        }\n        let orders = await this[method] (this.extend (request, params));\n        for (let i = 0; i < orders.length; i++) {\n            orders[i] = this.extend (orders[i], { 'status': 'open' });\n        }\n        return this.parseOrders (orders, market, since, limit);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetOrder (this.extend ({\n            'id': id.toString (),\n        }, params));\n        return this.parseOrder (response);\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let auth = nonce + this.uid + this.apiKey;\n            let signature = this.hmac (this.encode (auth), this.encode (this.secret));\n            body = this.urlencode (this.extend ({\n                'key': this.apiKey,\n                'signature': signature.toUpperCase (),\n                'nonce': nonce,\n            }, query));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (!response) {\n            throw new ExchangeError (this.id + ' returned ' + this.json (response));\n        } else if (response === true) {\n            return response;\n        } else if ('e' in response) {\n            if ('ok' in response)\n                if (response['ok'] === 'ok')\n                    return response;\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        } else if ('error' in response) {\n            if (response['error'])\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        return response;\n    }\n};\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst zb = require ('./zb.js');\nconst { ExchangeError } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class chbtc extends zb {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'chbtc',\n            'name': 'CHBTC',\n            'countries': 'CN',\n            'rateLimit': 1000,\n            'version': 'v1',\n            'hasCORS': false,\n            'hasFetchOrder': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28555659-f0040dc2-7109-11e7-9d99-688a438bf9f4.jpg',\n                'api': {\n                    'public': 'http://api.chbtc.com/data', // no https for public API\n                    'private': 'https://trade.chbtc.com/api',\n                },\n                'www': 'https://trade.chbtc.com/api',\n                'doc': 'https://www.chbtc.com/i/developer',\n            },\n        });\n    }\n\n    getMarketFieldName () {\n        return 'currency';\n    }\n\n    async fetchMarkets () {\n        return {\n            'BTC/CNY': { 'id': 'btc_cny', 'symbol': 'BTC/CNY', 'base': 'BTC', 'quote': 'CNY' },\n            'LTC/CNY': { 'id': 'ltc_cny', 'symbol': 'LTC/CNY', 'base': 'LTC', 'quote': 'CNY' },\n            'ETH/CNY': { 'id': 'eth_cny', 'symbol': 'ETH/CNY', 'base': 'ETH', 'quote': 'CNY' },\n            'ETC/CNY': { 'id': 'etc_cny', 'symbol': 'ETC/CNY', 'base': 'ETC', 'quote': 'CNY' },\n            'BTS/CNY': { 'id': 'bts_cny', 'symbol': 'BTS/CNY', 'base': 'BTS', 'quote': 'CNY' },\n            // 'EOS/CNY': { 'id': 'eos_cny', 'symbol': 'EOS/CNY', 'base': 'EOS', 'quote': 'CNY' },\n            'BCH/CNY': { 'id': 'bcc_cny', 'symbol': 'BCH/CNY', 'base': 'BCH', 'quote': 'CNY' },\n            'HSR/CNY': { 'id': 'hsr_cny', 'symbol': 'HSR/CNY', 'base': 'HSR', 'quote': 'CNY' },\n            'QTUM/CNY': { 'id': 'qtum_cny', 'symbol': 'QTUM/CNY', 'base': 'QTUM', 'quote': 'CNY' },\n        };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (api == 'private') {\n            if ('code' in response)\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        if ('result' in response) {\n            if (!response['result'])\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        return response;\n    }\n\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst foxbit = require ('./foxbit.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class chilebit extends foxbit {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'chilebit',\n            'name': 'ChileBit',\n            'countries': 'CL',\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27991414-1298f0d8-647f-11e7-9c40-d56409266336.jpg',\n                'api': {\n                    'public': 'https://api.blinktrade.com/api',\n                    'private': 'https://api.blinktrade.com/tapi',\n                },\n                'www': 'https://chilebit.net',\n                'doc': 'https://blinktrade.com/docs',\n            },\n        });\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, NotSupported } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class coincheck extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'coincheck',\n            'name': 'coincheck',\n            'countries': [ 'JP', 'ID' ],\n            'rateLimit': 1500,\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766464-3b5c3c74-5ed9-11e7-840e-31b32968e1da.jpg',\n                'api': 'https://coincheck.com/api',\n                'www': 'https://coincheck.com',\n                'doc': 'https://coincheck.com/documents/exchange/api',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'exchange/orders/rate',\n                        'order_books',\n                        'rate/{pair}',\n                        'ticker',\n                        'trades',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'accounts',\n                        'accounts/balance',\n                        'accounts/leverage_balance',\n                        'bank_accounts',\n                        'deposit_money',\n                        'exchange/orders/opens',\n                        'exchange/orders/transactions',\n                        'exchange/orders/transactions_pagination',\n                        'exchange/leverage/positions',\n                        'lending/borrows/matches',\n                        'send_money',\n                        'withdraws',\n                    ],\n                    'post': [\n                        'bank_accounts',\n                        'deposit_money/{id}/fast',\n                        'exchange/orders',\n                        'exchange/transfers/to_leverage',\n                        'exchange/transfers/from_leverage',\n                        'lending/borrows',\n                        'lending/borrows/{id}/repay',\n                        'send_money',\n                        'withdraws',\n                    ],\n                    'delete': [\n                        'bank_accounts/{id}',\n                        'exchange/orders/{id}',\n                        'withdraws/{id}',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/JPY': { 'id': 'btc_jpy', 'symbol': 'BTC/JPY', 'base': 'BTC', 'quote': 'JPY' }, // the only real pair\n                // 'ETH/JPY': { 'id': 'eth_jpy', 'symbol': 'ETH/JPY', 'base': 'ETH', 'quote': 'JPY' },\n                // 'ETC/JPY': { 'id': 'etc_jpy', 'symbol': 'ETC/JPY', 'base': 'ETC', 'quote': 'JPY' },\n                // 'DAO/JPY': { 'id': 'dao_jpy', 'symbol': 'DAO/JPY', 'base': 'DAO', 'quote': 'JPY' },\n                // 'LSK/JPY': { 'id': 'lsk_jpy', 'symbol': 'LSK/JPY', 'base': 'LSK', 'quote': 'JPY' },\n                // 'FCT/JPY': { 'id': 'fct_jpy', 'symbol': 'FCT/JPY', 'base': 'FCT', 'quote': 'JPY' },\n                // 'XMR/JPY': { 'id': 'xmr_jpy', 'symbol': 'XMR/JPY', 'base': 'XMR', 'quote': 'JPY' },\n                // 'REP/JPY': { 'id': 'rep_jpy', 'symbol': 'REP/JPY', 'base': 'REP', 'quote': 'JPY' },\n                // 'XRP/JPY': { 'id': 'xrp_jpy', 'symbol': 'XRP/JPY', 'base': 'XRP', 'quote': 'JPY' },\n                // 'ZEC/JPY': { 'id': 'zec_jpy', 'symbol': 'ZEC/JPY', 'base': 'ZEC', 'quote': 'JPY' },\n                // 'XEM/JPY': { 'id': 'xem_jpy', 'symbol': 'XEM/JPY', 'base': 'XEM', 'quote': 'JPY' },\n                // 'LTC/JPY': { 'id': 'ltc_jpy', 'symbol': 'LTC/JPY', 'base': 'LTC', 'quote': 'JPY' },\n                // 'DASH/JPY': { 'id': 'dash_jpy', 'symbol': 'DASH/JPY', 'base': 'DASH', 'quote': 'JPY' },\n                // 'ETH/BTC': { 'id': 'eth_btc', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC' },\n                // 'ETC/BTC': { 'id': 'etc_btc', 'symbol': 'ETC/BTC', 'base': 'ETC', 'quote': 'BTC' },\n                // 'LSK/BTC': { 'id': 'lsk_btc', 'symbol': 'LSK/BTC', 'base': 'LSK', 'quote': 'BTC' },\n                // 'FCT/BTC': { 'id': 'fct_btc', 'symbol': 'FCT/BTC', 'base': 'FCT', 'quote': 'BTC' },\n                // 'XMR/BTC': { 'id': 'xmr_btc', 'symbol': 'XMR/BTC', 'base': 'XMR', 'quote': 'BTC' },\n                // 'REP/BTC': { 'id': 'rep_btc', 'symbol': 'REP/BTC', 'base': 'REP', 'quote': 'BTC' },\n                // 'XRP/BTC': { 'id': 'xrp_btc', 'symbol': 'XRP/BTC', 'base': 'XRP', 'quote': 'BTC' },\n                // 'ZEC/BTC': { 'id': 'zec_btc', 'symbol': 'ZEC/BTC', 'base': 'ZEC', 'quote': 'BTC' },\n                // 'XEM/BTC': { 'id': 'xem_btc', 'symbol': 'XEM/BTC', 'base': 'XEM', 'quote': 'BTC' },\n                // 'LTC/BTC': { 'id': 'ltc_btc', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC' },\n                // 'DASH/BTC': { 'id': 'dash_btc', 'symbol': 'DASH/BTC', 'base': 'DASH', 'quote': 'BTC' },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let balances = await this.privateGetAccountsBalance ();\n        let result = { 'info': balances };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let account = this.account ();\n            if (lowercase in balances)\n                account['free'] = parseFloat (balances[lowercase]);\n            let reserved = lowercase + '_reserved';\n            if (reserved in balances)\n                account['used'] = parseFloat (balances[reserved]);\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        if (symbol != 'BTC/JPY')\n            throw new NotSupported (this.id + ' fetchOrderBook () supports BTC/JPY only');\n        let orderbook = await this.publicGetOrderBooks (params);\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        if (symbol != 'BTC/JPY')\n            throw new NotSupported (this.id + ' fetchTicker () supports BTC/JPY only');\n        let ticker = await this.publicGetTicker (params);\n        let timestamp = ticker['timestamp'] * 1000;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = this.parse8601 (trade['created_at']);\n        return {\n            'id': trade['id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['order_type'],\n            'price': parseFloat (trade['rate']),\n            'amount': parseFloat (trade['amount']),\n            'info': trade,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        if (symbol != 'BTC/JPY')\n            throw new NotSupported (this.id + ' fetchTrades () supports BTC/JPY only');\n        let market = this.market (symbol);\n        let response = await this.publicGetTrades (this.extend ({\n            'pair': market['id'],\n        }, params));\n        if ('success' in response)\n            if (response['success'])\n                if (typeof response['data'] !== 'undefined')\n                    return this.parseTrades (response['data'], market, since, limit);\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let order = {\n            'pair': this.marketId (symbol),\n        };\n        if (type == 'market') {\n            let order_type = type + '_' + side;\n            order['order_type'] = order_type;\n            let prefix = (side == 'buy') ? (order_type + '_') : '';\n            order[prefix + 'amount'] = amount;\n        } else {\n            order['order_type'] = side;\n            order['rate'] = price;\n            order['amount'] = amount;\n        }\n        let response = await this.privatePostExchangeOrders (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privateDeleteExchangeOrdersId ({ 'id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let queryString = '';\n            if (method == 'GET') {\n                if (Object.keys (query).length)\n                    url += '?' + this.urlencode (this.keysort (query));\n            } else {\n                if (Object.keys (query).length) {\n                    body = this.urlencode (this.keysort (query));\n                    queryString = body;\n                }\n            }\n            let auth = nonce + url + queryString;\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'ACCESS-KEY': this.apiKey,\n                'ACCESS-NONCE': nonce,\n                'ACCESS-SIGNATURE': this.hmac (this.encode (auth), this.encode (this.secret)),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (api == 'public')\n            return response;\n        if ('success' in response)\n            if (response['success'])\n                return response;\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class coinexchange extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'coinexchange',\n            'name': 'CoinExchange',\n            'countries': [ 'IN', 'JP', 'KR', 'VN', 'US' ],\n            'rateLimit': 1000,\n            // obsolete metainfo interface\n            'hasPrivateAPI': false,\n            'hasFetchTrades': false,\n            'hasFetchCurrencies': true,\n            'hasFetchTickers': true,\n            // new metainfo interface\n            'has': {\n                'fetchTrades': false,\n                'fetchCurrencies': true,\n                'fetchTickers': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/34842303-29c99fca-f71c-11e7-83c1-09d900cb2334.jpg',\n                'api': 'https://www.coinexchange.io/api/v1',\n                'www': 'https://www.coinexchange.io',\n                'doc': 'https://coinexchangeio.github.io/slate/',\n                'fees': 'https://www.coinexchange.io/fees',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'getcurrency',\n                        'getcurrencies',\n                        'getmarkets',\n                        'getmarketsummaries',\n                        'getmarketsummary',\n                        'getorderbook',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.0015,\n                    'taker': 0.0015,\n                },\n            },\n            'precision': {\n                'amount': 8,\n                'price': 8,\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        return currency;\n    }\n\n    async fetchCurrencies (params = {}) {\n        let currencies = await this.publicGetCurrencies (params);\n        let precision = this.precision['amount'];\n        let result = {};\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let id = currency['CurrencyID'];\n            let code = this.commonCurrencyCode (currency['TickerCode']);\n            let active = currency['WalletStatus'] == 'online';\n            let status = 'ok';\n            if (!active)\n                status = 'disabled';\n            result[code] = {\n                'id': id,\n                'code': code,\n                'name': currency['Name'],\n                'active': active,\n                'status': status,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': undefined,\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': undefined,\n                        'max': Math.pow (10, precision),\n                    },\n                },\n                'info': currency,\n            };\n        }\n        return result;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetMarkets ();\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let id = market['MarketID'];\n            let base = this.commonCurrencyCode (market['MarketAssetCode']);\n            let quote = this.commonCurrencyCode (market['BaseCurrencyCode']);\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': true,\n                'lot': undefined,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        if (!market) {\n            let marketId = ticker['MarketID'];\n            market = this.marketsById[marketId];\n        }\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['HighPrice']),\n            'low': parseFloat (ticker['LowPrice']),\n            'bid': parseFloat (ticker['BidPrice']),\n            'ask': parseFloat (ticker['AskPrice']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['LastPrice']),\n            'change': parseFloat (ticker['Change']),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': parseFloat (ticker['Volume']),\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetMarketsummary (this.extend ({\n            'market_id': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetMarketsummaries (params);\n        let result = {};\n        for (let i = 0; i < tickers.length; i++) {\n            let ticker = this.parseTicker (tickers[i]);\n            let symbol = ticker['symbol'];\n            result[symbol] = ticker;\n        }\n        return result;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderbook (this.extend ({\n            'market_id': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'BuyOrders', 'SellOrders', 'Price', 'Quantity');\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + path;\n        if (api == 'public') {\n            params = this.urlencode (params);\n            if (params.length)\n                url += '?' + params;\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        let success = this.safeInteger (response, 'success');\n        if (success != 1) {\n            throw new ExchangeError (response['message']);\n        }\n        return response['result'];\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class coinfloor extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'coinfloor',\n            'name': 'coinfloor',\n            'rateLimit': 1000,\n            'countries': 'UK',\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28246081-623fc164-6a1c-11e7-913f-bac0d5576c90.jpg',\n                'api': 'https://webapi.coinfloor.co.uk:8090/bist',\n                'www': 'https://www.coinfloor.co.uk',\n                'doc': [\n                    'https://github.com/coinfloor/api',\n                    'https://www.coinfloor.co.uk/api',\n                ],\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'uid': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        '{id}/ticker/',\n                        '{id}/order_book/',\n                        '{id}/transactions/',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        '{id}/balance/',\n                        '{id}/user_transactions/',\n                        '{id}/open_orders/',\n                        '{id}/cancel_order/',\n                        '{id}/buy/',\n                        '{id}/sell/',\n                        '{id}/buy_market/',\n                        '{id}/sell_market/',\n                        '{id}/estimate_sell_market/',\n                        '{id}/estimate_buy_market/',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/GBP': { 'id': 'XBT/GBP', 'symbol': 'BTC/GBP', 'base': 'BTC', 'quote': 'GBP' },\n                'BTC/EUR': { 'id': 'XBT/EUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR' },\n                'BTC/USD': { 'id': 'XBT/USD', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD' },\n                'BTC/PLN': { 'id': 'XBT/PLN', 'symbol': 'BTC/PLN', 'base': 'BTC', 'quote': 'PLN' },\n                'BCH/GBP': { 'id': 'BCH/GBP', 'symbol': 'BCH/GBP', 'base': 'BCH', 'quote': 'GBP' },\n            },\n        });\n    }\n\n    fetchBalance (params = {}) {\n        let symbol = undefined;\n        if ('symbol' in params)\n            symbol = params['symbol'];\n        if ('id' in params)\n            symbol = params['id'];\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchBalance requires a symbol param');\n        // todo parse balance\n        return this.privatePostIdBalance ({\n            'id': this.marketId (symbol),\n        });\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetIdOrderBook (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        // rewrite to get the timestamp from HTTP headers\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let vwap = this.safeFloat (ticker, 'vwap');\n        let baseVolume = parseFloat (ticker['volume']);\n        let quoteVolume = undefined;\n        if (typeof vwap !== 'undefined') {\n            quoteVolume = baseVolume * vwap;\n        }\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': vwap,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let market = this.market (symbol);\n        let ticker = await this.publicGetIdTicker (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['date'] * 1000;\n        return {\n            'info': trade,\n            'id': trade['tid'].toString (),\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetIdTransactions (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let order = { 'id': this.marketId (symbol) };\n        let method = 'privatePostId' + this.capitalize (side);\n        if (type == 'market') {\n            order['quantity'] = amount;\n            method += 'Market';\n        } else {\n            order['price'] = price;\n            order['amount'] = amount;\n        }\n        return this[method] (this.extend (order, params));\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostIdCancelOrder ({ 'id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        // curl -k -u '[User ID]/[API key]:[Passphrase]' https://webapi.coinfloor.co.uk:8090/bist/XBT/GBP/balance/\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({ 'nonce': nonce }, query));\n            let auth = this.uid + '/' + this.apiKey + ':' + this.password;\n            let signature = this.stringToBase64 (auth);\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Authorization': 'Basic ' + signature,\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class coingi extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'coingi',\n            'name': 'Coingi',\n            'rateLimit': 1000,\n            'countries': [ 'PA', 'BG', 'CN', 'US' ], // Panama, Bulgaria, China, US\n            'hasFetchTickers': true,\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28619707-5c9232a8-7212-11e7-86d6-98fe5d15cc6e.jpg',\n                'api': {\n                    'www': 'https://coingi.com',\n                    'current': 'https://api.coingi.com',\n                    'user': 'https://api.coingi.com',\n                },\n                'www': 'https://coingi.com',\n                'doc': 'http://docs.coingi.apiary.io/',\n            },\n            'api': {\n                'www': {\n                    'get': [\n                        '',\n                    ],\n                },\n                'current': {\n                    'get': [\n                        'order-book/{pair}/{askCount}/{bidCount}/{depth}',\n                        'transactions/{pair}/{maxCount}',\n                        '24hour-rolling-aggregation',\n                    ],\n                },\n                'user': {\n                    'post': [\n                        'balance',\n                        'add-order',\n                        'cancel-order',\n                        'orders',\n                        'transactions',\n                        'create-crypto-withdrawal',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'taker': 0.2 / 100,\n                    'maker': 0.2 / 100,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.001,\n                        'LTC': 0.01,\n                        'DOGE': 2,\n                        'PPC': 0.02,\n                        'VTC': 0.2,\n                        'NMC': 2,\n                        'DASH': 0.002,\n                        'USD': 10,\n                        'EUR': 10,\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                        'LTC': 0,\n                        'DOGE': 0,\n                        'PPC': 0,\n                        'VTC': 0,\n                        'NMC': 0,\n                        'DASH': 0,\n                        'USD': 5,\n                        'EUR': 1,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        this.parseJsonResponse = false;\n        let response = await this.wwwGet ();\n        this.parseJsonResponse = true;\n        let parts = response.split ('do=currencyPairSelector-selectCurrencyPair\" class=\"active\">');\n        let currencyParts = parts[1].split ('<div class=\"currency-pair-label\">');\n        let result = [];\n        for (let i = 1; i < currencyParts.length; i++) {\n            let currencyPart = currencyParts[i];\n            let idParts = currencyPart.split ('</div>');\n            let id = idParts[0];\n            let symbol = id;\n            id = id.replace ('/', '-');\n            id = id.toLowerCase ();\n            let [ base, quote ] = symbol.split ('/');\n            let precision = {\n                'amount': 8,\n                'price': 8,\n            };\n            let lot = Math.pow (10, -precision['amount']);\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': id,\n                'lot': lot,\n                'active': true,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': Math.pow (10, precision['amount']),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision['price']),\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': 0,\n                        'max': undefined,\n                    },\n                },\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let lowercaseCurrencies = [];\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            lowercaseCurrencies.push (currency.toLowerCase ());\n        }\n        let balances = await this.userPostBalance ({\n            'currencies': lowercaseCurrencies.join (','),\n        });\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency']['name'];\n            currency = currency.toUpperCase ();\n            let account = {\n                'free': balance['available'],\n                'used': balance['blocked'] + balance['inOrders'] + balance['withdrawing'],\n                'total': 0.0,\n            };\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orderbook = await this.currentGetOrderBookPairAskCountBidCountDepth (this.extend ({\n            'pair': market['id'],\n            'askCount': 512, // maximum returned number of asks 1-512\n            'bidCount': 512, // maximum returned number of bids 1-512\n            'depth': 32, // maximum number of depth range steps 1-32\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'baseAmount');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': ticker['high'],\n            'low': ticker['low'],\n            'bid': ticker['highestBid'],\n            'ask': ticker['lowestAsk'],\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': undefined,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': ticker['baseVolume'],\n            'quoteVolume': ticker['counterVolume'],\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.currentGet24hourRollingAggregation (params);\n        let result = {};\n        for (let t = 0; t < response.length; t++) {\n            let ticker = response[t];\n            let base = ticker['currencyPair']['base'].toUpperCase ();\n            let quote = ticker['currencyPair']['counter'].toUpperCase ();\n            let symbol = base + '/' + quote;\n            let market = undefined;\n            if (symbol in this.markets) {\n                market = this.markets[symbol];\n            }\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.fetchTickers (undefined, params);\n        if (symbol in tickers)\n            return tickers[symbol];\n        throw new ExchangeError (this.id + ' return did not contain ' + symbol);\n    }\n\n    parseTrade (trade, market = undefined) {\n        if (!market)\n            market = this.markets_by_id[trade['currencyPair']];\n        return {\n            'id': trade['id'],\n            'info': trade,\n            'timestamp': trade['timestamp'],\n            'datetime': this.iso8601 (trade['timestamp']),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined, // type\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.currentGetTransactionsPairMaxCount (this.extend ({\n            'pair': market['id'],\n            'maxCount': 128,\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'currencyPair': this.marketId (symbol),\n            'volume': amount,\n            'price': price,\n            'orderType': (side == 'buy') ? 0 : 1,\n        };\n        let response = await this.userPostAddOrder (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['result'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.userPostCancelOrder ({ 'orderId': id });\n    }\n\n    sign (path, api = 'current', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        if (api != 'www') {\n            url += '/' + api + '/' + this.implodeParams (path, params);\n        }\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'current') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else if (api == 'user') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let request = this.extend ({\n                'token': this.apiKey,\n                'nonce': nonce,\n            }, query);\n            let auth = nonce.toString () + '$' + this.apiKey;\n            request['signature'] = this.hmac (this.encode (auth), this.encode (this.secret));\n            body = this.json (request);\n            headers = {\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'current', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (typeof response !== 'string') {\n            if ('errors' in response)\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class coinmarketcap extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'coinmarketcap',\n            'name': 'CoinMarketCap',\n            'rateLimit': 10000,\n            'version': 'v1',\n            'countries': 'US',\n            'hasCORS': true,\n            'hasPrivateAPI': false,\n            'hasCreateOrder': false,\n            'hasCancelOrder': false,\n            'hasFetchBalance': false,\n            'hasFetchOrderBook': false,\n            'hasFetchTrades': false,\n            'hasFetchTickers': true,\n            'hasFetchCurrencies': true,\n            'has': {\n                'fetchCurrencies': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28244244-9be6312a-69ed-11e7-99c1-7c1797275265.jpg',\n                'api': {\n                    'public': 'https://api.coinmarketcap.com',\n                    'files': 'https://files.coinmarketcap.com',\n                    'charts': 'https://graph.coinmarketcap.com',\n                },\n                'www': 'https://coinmarketcap.com',\n                'doc': 'https://coinmarketcap.com/api',\n            },\n            'requiredCredentials': {\n                'apiKey': false,\n                'secret': false,\n            },\n            'api': {\n                'files': {\n                    'get': [\n                        'generated/stats/global.json',\n                    ],\n                },\n                'graphs': {\n                    'get': [\n                        'currencies/{name}/',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'ticker/',\n                        'ticker/{id}/',\n                        'global/',\n                    ],\n                },\n            },\n            'currencyCodes': [\n                'AUD',\n                'BRL',\n                'CAD',\n                'CHF',\n                'CNY',\n                'EUR',\n                'GBP',\n                'HKD',\n                'IDR',\n                'INR',\n                'JPY',\n                'KRW',\n                'MXN',\n                'RUB',\n                'USD',\n            ],\n        });\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        throw new ExchangeError ('Fetching order books is not supported by the API of ' + this.id);\n    }\n\n    currencyCode (base, name) {\n        const currencies = {\n            'Bitgem': 'Bitgem',\n            'NetCoin': 'NetCoin',\n            'BatCoin': 'BatCoin',\n        };\n        if (name in currencies)\n            return currencies[name];\n        return base;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetTicker ({\n            'limit': 0,\n        });\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let currencies = this.currencyCodes;\n            for (let i = 0; i < currencies.length; i++) {\n                let quote = currencies[i];\n                let quoteId = quote.toLowerCase ();\n                let baseId = market['id'];\n                let base = this.currencyCode (market['symbol'], market['name']);\n                let symbol = base + '/' + quote;\n                let id = baseId + '/' + quote;\n                result.push ({\n                    'id': id,\n                    'symbol': symbol,\n                    'base': base,\n                    'quote': quote,\n                    'baseId': baseId,\n                    'quoteId': quoteId,\n                    'info': market,\n                });\n            }\n        }\n        return result;\n    }\n\n    async fetchGlobal (currency = 'USD') {\n        await this.loadMarkets ();\n        let request = {};\n        if (currency)\n            request['convert'] = currency;\n        return await this.publicGetGlobal (request);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        if ('last_updated' in ticker)\n            if (ticker['last_updated'])\n                timestamp = parseInt (ticker['last_updated']) * 1000;\n        let change = undefined;\n        if ('percent_change_24h' in ticker)\n            if (ticker['percent_change_24h'])\n                change = this.safeFloat (ticker, 'percent_change_24h');\n        let last = undefined;\n        let symbol = undefined;\n        let volume = undefined;\n        if (market) {\n            let priceKey = 'price_' + market['quoteId'];\n            if (priceKey in ticker)\n                if (ticker[priceKey])\n                    last = this.safeFloat (ticker, priceKey);\n            symbol = market['symbol'];\n            let volumeKey = '24h_volume_' + market['quoteId'];\n            if (volumeKey in ticker)\n                if (ticker[volumeKey])\n                    volume = this.safeFloat (ticker, volumeKey);\n        }\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': undefined,\n            'ask': undefined,\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': last,\n            'change': change,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': volume,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (currency = 'USD', params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            'limit': 10000,\n        };\n        if (currency)\n            request['convert'] = currency;\n        let response = await this.publicGetTicker (this.extend (request, params));\n        let tickers = {};\n        for (let t = 0; t < response.length; t++) {\n            let ticker = response[t];\n            let id = ticker['id'] + '/' + currency;\n            let symbol = id;\n            let market = undefined;\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            }\n            tickers[symbol] = this.parseTicker (ticker, market);\n        }\n        return tickers;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = this.extend ({\n            'convert': market['quote'],\n            'id': market['baseId'],\n        }, params);\n        let response = await this.publicGetTickerId (request);\n        let ticker = response[0];\n        return this.parseTicker (ticker, market);\n    }\n\n    async fetchCurrencies (params = {}) {\n        let currencies = await this.publicGetTicker (this.extend ({\n            'limit': 0,\n        }, params));\n        let result = {};\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let id = currency['symbol'];\n            let name = currency['name'];\n            // todo: will need to rethink the fees\n            // to add support for multiple withdrawal/deposit methods and\n            // differentiated fees for each particular method\n            let precision = 8; // default precision, todo: fix \"magic constants\"\n            let code = this.currencyCode (id, name);\n            result[code] = {\n                'id': id,\n                'code': code,\n                'info': currency,\n                'name': name,\n                'active': true,\n                'status': 'ok',\n                'fee': undefined, // todo: redesign\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                },\n            };\n        }\n        return result;\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api] + '/' + this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (Object.keys (query).length)\n            url += '?' + this.urlencode (query);\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response) {\n            if (response['error']) {\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n            }\n        }\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class coinmate extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'coinmate',\n            'name': 'CoinMate',\n            'countries': [ 'GB', 'CZ', 'EU' ], // UK, Czech Republic\n            'rateLimit': 1000,\n            'hasCORS': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27811229-c1efb510-606c-11e7-9a36-84ba2ce412d8.jpg',\n                'api': 'https://coinmate.io/api',\n                'www': 'https://coinmate.io',\n                'doc': [\n                    'http://docs.coinmate.apiary.io',\n                    'https://coinmate.io/developers',\n                ],\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'uid': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'orderBook',\n                        'ticker',\n                        'transactions',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balances',\n                        'bitcoinWithdrawal',\n                        'bitcoinDepositAddresses',\n                        'buyInstant',\n                        'buyLimit',\n                        'cancelOrder',\n                        'cancelOrderWithInfo',\n                        'createVoucher',\n                        'openOrders',\n                        'redeemVoucher',\n                        'sellInstant',\n                        'sellLimit',\n                        'transactionHistory',\n                        'unconfirmedBitcoinDeposits',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/EUR': { 'id': 'BTC_EUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR', 'precision': { 'amount': 4, 'price': 2 }},\n                'BTC/CZK': { 'id': 'BTC_CZK', 'symbol': 'BTC/CZK', 'base': 'BTC', 'quote': 'CZK', 'precision': { 'amount': 4, 'price': 2 }},\n                'LTC/BTC': { 'id': 'LTC_BTC', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC', 'precision': { 'amount': 4, 'price': 5 }},\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.0005,\n                    'taker': 0.0035,\n                },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostBalances ();\n        let balances = response['data'];\n        let result = { 'info': balances };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            if (currency in balances) {\n                account['free'] = balances[currency]['available'];\n                account['used'] = balances[currency]['reserved'];\n                account['total'] = balances[currency]['balance'];\n            }\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let response = await this.publicGetOrderBook (this.extend ({\n            'currencyPair': this.marketId (symbol),\n            'groupByPriceLimit': 'False',\n        }, params));\n        let orderbook = response['data'];\n        let timestamp = orderbook['timestamp'] * 1000;\n        return this.parseOrderBook (orderbook, timestamp, 'bids', 'asks', 'price', 'amount');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let response = await this.publicGetTicker (this.extend ({\n            'currencyPair': this.marketId (symbol),\n        }, params));\n        let ticker = response['data'];\n        let timestamp = ticker['timestamp'] * 1000;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['amount']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        if (!market)\n            market = this.markets_by_id[trade['currencyPair']];\n        return {\n            'id': trade['transactionId'],\n            'info': trade,\n            'timestamp': trade['timestamp'],\n            'datetime': this.iso8601 (trade['timestamp']),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetTransactions (this.extend ({\n            'currencyPair': market['id'],\n            'minutesIntoHistory': 10,\n        }, params));\n        return this.parseTrades (response['data'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let method = 'privatePost' + this.capitalize (side);\n        let order = {\n            'currencyPair': this.marketId (symbol),\n        };\n        if (type == 'market') {\n            if (side == 'buy')\n                order['total'] = amount; // amount in fiat\n            else\n                order['amount'] = amount; // amount in fiat\n            method += 'Instant';\n        } else {\n            order['amount'] = amount; // amount in crypto\n            order['price'] = price;\n            method += this.capitalize (type);\n        }\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['data'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelOrder ({ 'orderId': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + path;\n        if (api == 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let auth = nonce + this.uid + this.apiKey;\n            let signature = this.hmac (this.encode (auth), this.encode (this.secret));\n            body = this.urlencode (this.extend ({\n                'clientId': this.uid,\n                'nonce': nonce,\n                'publicKey': this.apiKey,\n                'signature': signature.toUpperCase (),\n            }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response)\n            if (response['error'])\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class coinsecure extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'coinsecure',\n            'name': 'Coinsecure',\n            'countries': 'IN', // India\n            'rateLimit': 1000,\n            'version': 'v1',\n            'hasCORS': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766472-9cbd200a-5ed9-11e7-9551-2267ad7bac08.jpg',\n                'api': 'https://api.coinsecure.in',\n                'www': 'https://coinsecure.in',\n                'doc': [\n                    'https://api.coinsecure.in',\n                    'https://github.com/coinsecure/plugins',\n                ],\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': false,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'bitcoin/search/confirmation/{txid}',\n                        'exchange/ask/low',\n                        'exchange/ask/orders',\n                        'exchange/bid/high',\n                        'exchange/bid/orders',\n                        'exchange/lastTrade',\n                        'exchange/max24Hr',\n                        'exchange/min24Hr',\n                        'exchange/ticker',\n                        'exchange/trades',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'mfa/authy/call',\n                        'mfa/authy/sms',\n                        'netki/search/{netkiName}',\n                        'user/bank/otp/{number}',\n                        'user/kyc/otp/{number}',\n                        'user/profile/phone/otp/{number}',\n                        'user/wallet/coin/address/{id}',\n                        'user/wallet/coin/deposit/confirmed/all',\n                        'user/wallet/coin/deposit/confirmed/{id}',\n                        'user/wallet/coin/deposit/unconfirmed/all',\n                        'user/wallet/coin/deposit/unconfirmed/{id}',\n                        'user/wallet/coin/wallets',\n                        'user/exchange/bank/fiat/accounts',\n                        'user/exchange/bank/fiat/balance/available',\n                        'user/exchange/bank/fiat/balance/pending',\n                        'user/exchange/bank/fiat/balance/total',\n                        'user/exchange/bank/fiat/deposit/cancelled',\n                        'user/exchange/bank/fiat/deposit/unverified',\n                        'user/exchange/bank/fiat/deposit/verified',\n                        'user/exchange/bank/fiat/withdraw/cancelled',\n                        'user/exchange/bank/fiat/withdraw/completed',\n                        'user/exchange/bank/fiat/withdraw/unverified',\n                        'user/exchange/bank/fiat/withdraw/verified',\n                        'user/exchange/ask/cancelled',\n                        'user/exchange/ask/completed',\n                        'user/exchange/ask/pending',\n                        'user/exchange/bid/cancelled',\n                        'user/exchange/bid/completed',\n                        'user/exchange/bid/pending',\n                        'user/exchange/bank/coin/addresses',\n                        'user/exchange/bank/coin/balance/available',\n                        'user/exchange/bank/coin/balance/pending',\n                        'user/exchange/bank/coin/balance/total',\n                        'user/exchange/bank/coin/deposit/cancelled',\n                        'user/exchange/bank/coin/deposit/unverified',\n                        'user/exchange/bank/coin/deposit/verified',\n                        'user/exchange/bank/coin/withdraw/cancelled',\n                        'user/exchange/bank/coin/withdraw/completed',\n                        'user/exchange/bank/coin/withdraw/unverified',\n                        'user/exchange/bank/coin/withdraw/verified',\n                        'user/exchange/bank/summary',\n                        'user/exchange/coin/fee',\n                        'user/exchange/fiat/fee',\n                        'user/exchange/kycs',\n                        'user/exchange/referral/coin/paid',\n                        'user/exchange/referral/coin/successful',\n                        'user/exchange/referral/fiat/paid',\n                        'user/exchange/referrals',\n                        'user/exchange/trade/summary',\n                        'user/login/token/{token}',\n                        'user/summary',\n                        'user/wallet/summary',\n                        'wallet/coin/withdraw/cancelled',\n                        'wallet/coin/withdraw/completed',\n                        'wallet/coin/withdraw/unverified',\n                        'wallet/coin/withdraw/verified',\n                    ],\n                    'post': [\n                        'login',\n                        'login/initiate',\n                        'login/password/forgot',\n                        'mfa/authy/initiate',\n                        'mfa/ga/initiate',\n                        'signup',\n                        'user/netki/update',\n                        'user/profile/image/update',\n                        'user/exchange/bank/coin/withdraw/initiate',\n                        'user/exchange/bank/coin/withdraw/newVerifycode',\n                        'user/exchange/bank/fiat/withdraw/initiate',\n                        'user/exchange/bank/fiat/withdraw/newVerifycode',\n                        'user/password/change',\n                        'user/password/reset',\n                        'user/wallet/coin/withdraw/initiate',\n                        'wallet/coin/withdraw/newVerifycode',\n                    ],\n                    'put': [\n                        'signup/verify/{token}',\n                        'user/exchange/kyc',\n                        'user/exchange/bank/fiat/deposit/new',\n                        'user/exchange/ask/new',\n                        'user/exchange/bid/new',\n                        'user/exchange/instant/buy',\n                        'user/exchange/instant/sell',\n                        'user/exchange/bank/coin/withdraw/verify',\n                        'user/exchange/bank/fiat/account/new',\n                        'user/exchange/bank/fiat/withdraw/verify',\n                        'user/mfa/authy/initiate/enable',\n                        'user/mfa/ga/initiate/enable',\n                        'user/netki/create',\n                        'user/profile/phone/new',\n                        'user/wallet/coin/address/new',\n                        'user/wallet/coin/new',\n                        'user/wallet/coin/withdraw/sendToExchange',\n                        'user/wallet/coin/withdraw/verify',\n                    ],\n                    'delete': [\n                        'user/gcm/{code}',\n                        'user/logout',\n                        'user/exchange/bank/coin/withdraw/unverified/cancel/{withdrawID}',\n                        'user/exchange/bank/fiat/deposit/cancel/{depositID}',\n                        'user/exchange/ask/cancel/{orderID}',\n                        'user/exchange/bid/cancel/{orderID}',\n                        'user/exchange/bank/fiat/withdraw/unverified/cancel/{withdrawID}',\n                        'user/mfa/authy/disable/{code}',\n                        'user/mfa/ga/disable/{code}',\n                        'user/profile/phone/delete',\n                        'user/profile/image/delete/{netkiName}',\n                        'user/wallet/coin/withdraw/unverified/cancel/{withdrawID}',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/INR': { 'id': 'BTC/INR', 'symbol': 'BTC/INR', 'base': 'BTC', 'quote': 'INR' },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.4 / 100,\n                    'taker': 0.4 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privateGetUserExchangeBankSummary ();\n        let balance = response['message'];\n        let coin = {\n            'free': balance['availableCoinBalance'],\n            'used': balance['pendingCoinBalance'],\n            'total': balance['totalCoinBalance'],\n        };\n        let fiat = {\n            'free': balance['availableFiatBalance'],\n            'used': balance['pendingFiatBalance'],\n            'total': balance['totalFiatBalance'],\n        };\n        let result = {\n            'info': balance,\n            'BTC': coin,\n            'INR': fiat,\n        };\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let bids = await this.publicGetExchangeBidOrders (params);\n        let asks = await this.publicGetExchangeAskOrders (params);\n        let orderbook = {\n            'bids': bids['message'],\n            'asks': asks['message'],\n        };\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'rate', 'vol');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let response = await this.publicGetExchangeTicker (params);\n        let ticker = response['message'];\n        let timestamp = ticker['timestamp'];\n        let baseVolume = parseFloat (ticker['coinvolume']);\n        if (symbol == 'BTC/INR') {\n            let satoshi = 0.00000001;\n            baseVolume = baseVolume * satoshi;\n        }\n        let quoteVolume = parseFloat (ticker['fiatvolume']) / 100;\n        let vwap = quoteVolume / baseVolume;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']) / 100,\n            'low': parseFloat (ticker['low']) / 100,\n            'bid': parseFloat (ticker['bid']) / 100,\n            'ask': parseFloat (ticker['ask']) / 100,\n            'vwap': vwap,\n            'open': parseFloat (ticker['open']) / 100,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['lastPrice']) / 100,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, symbol = undefined) {\n        let timestamp = trade['time'];\n        let side = (trade['ordType'] == 'bid') ? 'buy' : 'sell';\n        return {\n            'id': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'order': undefined,\n            'symbol': symbol,\n            'type': undefined,\n            'side': side,\n            'price': this.safeFloat (trade, 'rate') / 100,\n            'amount': this.safeFloat (trade, 'vol') / 100000000,\n            'fee': undefined,\n            'info': trade,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let result = await this.publicGetExchangeTrades (params);\n        if ('message' in result) {\n            let trades = result['message'];\n            return this.parseTrades (trades, symbol);\n        }\n    }\n\n    async createOrder (market, type, side, amount, price = undefined, params = {}) {\n        let method = 'privatePutUserExchange';\n        let order = {};\n        if (type == 'market') {\n            method += 'Instant' + this.capitalize (side);\n            if (side == 'buy')\n                order['maxFiat'] = amount;\n            else\n                order['maxVol'] = amount;\n        } else {\n            let direction = (side == 'buy') ? 'Bid' : 'Ask';\n            method += direction + 'New';\n            order['rate'] = price;\n            order['vol'] = amount;\n        }\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['message']['orderID'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        throw new ExchangeError (this.id + ' cancelOrder () is not fully implemented yet');\n        let method = 'privateDeleteUserExchangeAskCancelOrderId'; // TODO fixme, have to specify order side here\n        return await this[method] ({ 'orderID': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            headers = { 'Authorization': this.apiKey };\n            if (Object.keys (query).length) {\n                body = this.json (query);\n                headers['Content-Type'] = 'application/json';\n            }\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code == 200) {\n            if ((body[0] == '{') || (body[0] == '[')) {\n                let response = JSON.parse (body);\n                if ('success' in response) {\n                    let success = response['success'];\n                    if (!success) {\n                        throw new ExchangeError (this.id + ' error returned: ' + body);\n                    }\n                    if (!('message' in response)) {\n                        throw new ExchangeError (this.id + ' malformed response: no \"message\" in response: ' + body);\n                    }\n                } else {\n                    throw new ExchangeError (this.id + ' malformed response: no \"success\" in response: ' + body);\n                }\n            } else {\n                // if not a JSON response\n                throw new ExchangeError (this.id + ' returned a non-JSON reply: ' + body);\n            }\n        }\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, AuthenticationError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class coinspot extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'coinspot',\n            'name': 'CoinSpot',\n            'countries': 'AU', // Australia\n            'rateLimit': 1000,\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28208429-3cacdf9a-6896-11e7-854e-4c79a772a30f.jpg',\n                'api': {\n                    'public': 'https://www.coinspot.com.au/pubapi',\n                    'private': 'https://www.coinspot.com.au/api',\n                },\n                'www': 'https://www.coinspot.com.au',\n                'doc': 'https://www.coinspot.com.au/api',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'latest',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'orders',\n                        'orders/history',\n                        'my/coin/deposit',\n                        'my/coin/send',\n                        'quote/buy',\n                        'quote/sell',\n                        'my/balances',\n                        'my/orders',\n                        'my/buy',\n                        'my/sell',\n                        'my/buy/cancel',\n                        'my/sell/cancel',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/AUD': { 'id': 'BTC', 'symbol': 'BTC/AUD', 'base': 'BTC', 'quote': 'AUD' },\n                'LTC/AUD': { 'id': 'LTC', 'symbol': 'LTC/AUD', 'base': 'LTC', 'quote': 'AUD' },\n                'DOGE/AUD': { 'id': 'DOGE', 'symbol': 'DOGE/AUD', 'base': 'DOGE', 'quote': 'AUD' },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostMyBalances ();\n        let result = { 'info': response };\n        if ('balance' in response) {\n            let balances = response['balance'];\n            let currencies = Object.keys (balances);\n            for (let c = 0; c < currencies.length; c++) {\n                let currency = currencies[c];\n                let uppercase = currency.toUpperCase ();\n                let account = {\n                    'free': balances[currency],\n                    'used': 0.0,\n                    'total': balances[currency],\n                };\n                if (uppercase == 'DRK')\n                    uppercase = 'DASH';\n                result[uppercase] = account;\n            }\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let market = this.market (symbol);\n        let orderbook = await this.privatePostOrders (this.extend ({\n            'cointype': market['id'],\n        }, params));\n        let result = this.parseOrderBook (orderbook, undefined, 'buyorders', 'sellorders', 'rate', 'amount');\n        result['bids'] = this.sortBy (result['bids'], 0, true);\n        result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let response = await this.publicGetLatest (params);\n        let id = this.marketId (symbol);\n        id = id.toLowerCase ();\n        let ticker = response['prices'][id];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        return this.privatePostOrdersHistory (this.extend ({\n            'cointype': this.marketId (symbol),\n        }, params));\n    }\n\n    createOrder (market, type, side, amount, price = undefined, params = {}) {\n        let method = 'privatePostMy' + this.capitalize (side);\n        if (type == 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let order = {\n            'cointype': this.marketId (market),\n            'amount': amount,\n            'rate': price,\n        };\n        return this[method] (this.extend (order, params));\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        throw new ExchangeError (this.id + ' cancelOrder () is not fully implemented yet');\n        let method = 'privatePostMyBuy';\n        return await this[method] ({ 'id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        if (!this.apiKey)\n            throw new AuthenticationError (this.id + ' requires apiKey for all requests');\n        let url = this.urls['api'][api] + '/' + path;\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.json (this.extend ({ 'nonce': nonce }, params));\n            headers = {\n                'Content-Type': 'application/json',\n                'key': this.apiKey,\n                'sign': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InsufficientFunds, OrderNotFound, OrderNotCached } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class cryptopia extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'cryptopia',\n            'name': 'Cryptopia',\n            'rateLimit': 1500,\n            'countries': 'NZ', // New Zealand\n            'hasCORS': false,\n            // obsolete metainfo interface\n            'hasFetchTickers': true,\n            'hasFetchOrder': true,\n            'hasFetchOrders': true,\n            'hasFetchOpenOrders': true,\n            'hasFetchClosedOrders': true,\n            'hasFetchMyTrades': true,\n            'hasFetchCurrencies': true,\n            'hasDeposit': true,\n            'hasWithdraw': true,\n            // new metainfo interface\n            'has': {\n                'fetchTickers': true,\n                'fetchOrder': 'emulated',\n                'fetchOrders': 'emulated',\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': 'emulated',\n                'fetchMyTrades': true,\n                'fetchCurrencies': true,\n                'deposit': true,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/29484394-7b4ea6e2-84c6-11e7-83e5-1fccf4b2dc81.jpg',\n                'api': 'https://www.cryptopia.co.nz/api',\n                'www': 'https://www.cryptopia.co.nz',\n                'doc': [\n                    'https://www.cryptopia.co.nz/Forum/Category/45',\n                    'https://www.cryptopia.co.nz/Forum/Thread/255',\n                    'https://www.cryptopia.co.nz/Forum/Thread/256',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'GetCurrencies',\n                        'GetTradePairs',\n                        'GetMarkets',\n                        'GetMarkets/{id}',\n                        'GetMarkets/{hours}',\n                        'GetMarkets/{id}/{hours}',\n                        'GetMarket/{id}',\n                        'GetMarket/{id}/{hours}',\n                        'GetMarketHistory/{id}',\n                        'GetMarketHistory/{id}/{hours}',\n                        'GetMarketOrders/{id}',\n                        'GetMarketOrders/{id}/{count}',\n                        'GetMarketOrderGroups/{ids}/{count}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'CancelTrade',\n                        'GetBalance',\n                        'GetDepositAddress',\n                        'GetOpenOrders',\n                        'GetTradeHistory',\n                        'GetTransactions',\n                        'SubmitTip',\n                        'SubmitTrade',\n                        'SubmitTransfer',\n                        'SubmitWithdraw',\n                    ],\n                },\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        const currencies = {\n            'ACC': 'AdCoin',\n            'CC': 'CCX',\n            'CMT': 'Comet',\n            'FCN': 'Facilecoin',\n            'NET': 'NetCoin',\n            'BTG': 'Bitgem',\n            'FUEL': 'FC2', // FuelCoin != FUEL\n            'QBT': 'Cubits',\n            'WRC': 'WarCoin',\n        };\n        if (currency in currencies)\n            return currencies[currency];\n        return currency;\n    }\n\n    currencyId (currency) {\n        const currencies = {\n            'AdCoin': 'ACC',\n            'CCX': 'CC',\n            'Comet': 'CMT',\n            'Cubits': 'QBT',\n            'Facilecoin': 'FCN',\n            'NetCoin': 'NET',\n            'Bitgem': 'BTG',\n            'FC2': 'FUEL',\n        };\n        if (currency in currencies)\n            return currencies[currency];\n        return currency;\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicGetTradePairs ();\n        let result = [];\n        let markets = response['Data'];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let id = market['Id'];\n            let symbol = market['Label'];\n            let [ base, quote ] = symbol.split ('/');\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            symbol = base + '/' + quote;\n            let precision = {\n                'amount': 8,\n                'price': 8,\n            };\n            let lot = market['MinimumTrade'];\n            let priceLimits = {\n                'min': market['MinimumPrice'],\n                'max': market['MaximumPrice'],\n            };\n            let amountLimits = {\n                'min': lot,\n                'max': market['MaximumTrade'],\n            };\n            let limits = {\n                'amount': amountLimits,\n                'price': priceLimits,\n                'cost': {\n                    'min': priceLimits['min'] * amountLimits['min'],\n                    'max': undefined,\n                },\n            };\n            let active = market['Status'] === 'OK';\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n                'maker': market['TradeFee'] / 100,\n                'taker': market['TradeFee'] / 100,\n                'lot': limits['amount']['min'],\n                'active': active,\n                'precision': precision,\n                'limits': limits,\n            });\n        }\n        return result;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetMarketOrdersId (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        let orderbook = response['Data'];\n        return this.parseOrderBook (orderbook, undefined, 'Buy', 'Sell', 'Price', 'Volume');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'info': ticker,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['High']),\n            'low': parseFloat (ticker['Low']),\n            'bid': parseFloat (ticker['BidPrice']),\n            'ask': parseFloat (ticker['AskPrice']),\n            'vwap': undefined,\n            'open': parseFloat (ticker['Open']),\n            'close': parseFloat (ticker['Close']),\n            'first': undefined,\n            'last': parseFloat (ticker['LastPrice']),\n            'change': parseFloat (ticker['Change']),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['Volume']),\n            'quoteVolume': parseFloat (ticker['BaseVolume']),\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetMarketId (this.extend ({\n            'id': market['id'],\n        }, params));\n        let ticker = response['Data'];\n        return this.parseTicker (ticker, market);\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetMarkets (params);\n        let result = {};\n        let tickers = response['Data'];\n        for (let i = 0; i < tickers.length; i++) {\n            let ticker = tickers[i];\n            let id = ticker['TradePairId'];\n            let recognized = (id in this.markets_by_id);\n            if (!recognized)\n                throw new ExchangeError (this.id + ' fetchTickers() returned unrecognized pair id ' + id);\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = undefined;\n        if ('Timestamp' in trade) {\n            timestamp = trade['Timestamp'] * 1000;\n        } else if ('TimeStamp' in trade) {\n            timestamp = this.parse8601 (trade['TimeStamp']);\n        }\n        let price = this.safeFloat (trade, 'Price');\n        if (!price)\n            price = this.safeFloat (trade, 'Rate');\n        let cost = this.safeFloat (trade, 'Total');\n        let id = this.safeString (trade, 'TradeId');\n        if (!market) {\n            if ('TradePairId' in trade)\n                if (trade['TradePairId'] in this.markets_by_id)\n                    market = this.markets_by_id[trade['TradePairId']];\n        }\n        let symbol = undefined;\n        let fee = undefined;\n        if (market) {\n            symbol = market['symbol'];\n            if ('Fee' in trade) {\n                fee = {\n                    'currency': market['quote'],\n                    'cost': trade['Fee'],\n                };\n            }\n        }\n        return {\n            'id': id,\n            'info': trade,\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': 'limit',\n            'side': trade['Type'].toLowerCase (),\n            'price': price,\n            'cost': cost,\n            'amount': trade['Amount'],\n            'fee': fee,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let hours = 24; // the default\n        if (since) {\n            let elapsed = this.milliseconds () - since;\n            let hour = 1000 * 60 * 60;\n            hours = parseInt (elapsed / hour);\n        }\n        let request = {\n            'id': market['id'],\n            'hours': hours,\n        };\n        let response = await this.publicGetMarketHistoryIdHours (this.extend (request, params));\n        let trades = response['Data'];\n        return this.parseTrades (trades, market, since, limit);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['TradePairId'] = market['id'];\n        }\n        let response = await this.privatePostGetTradeHistory (this.extend (request, params));\n        return this.parseTrades (response['Data'], market, since, limit);\n    }\n\n    async fetchCurrencies (params = {}) {\n        let response = await this.publicGetCurrencies (params);\n        let currencies = response['Data'];\n        let result = {};\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let id = currency['Symbol'];\n            // todo: will need to rethink the fees\n            // to add support for multiple withdrawal/deposit methods and\n            // differentiated fees for each particular method\n            let precision = 8; // default precision, todo: fix \"magic constants\"\n            let code = this.commonCurrencyCode (id);\n            let active = (currency['ListingStatus'] === 'Active');\n            let status = currency['Status'].toLowerCase ();\n            if (status !== 'ok')\n                active = false;\n            result[code] = {\n                'id': id,\n                'code': code,\n                'info': currency,\n                'name': currency['Name'],\n                'active': active,\n                'status': status,\n                'fee': currency['WithdrawFee'],\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': currency['MinBaseTrade'],\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': currency['MinWithdraw'],\n                        'max': currency['MaxWithdraw'],\n                    },\n                },\n            };\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetBalance ();\n        let balances = response['Data'];\n        let result = { 'info': response };\n        for (let i = 0; i < balances.length; i++) {\n            let balance = balances[i];\n            let code = balance['Symbol'];\n            let currency = this.commonCurrencyCode (code);\n            let account = {\n                'free': balance['Available'],\n                'used': 0.0,\n                'total': balance['Total'],\n            };\n            account['used'] = account['total'] - account['free'];\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type === 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        // price = parseFloat (price);\n        // amount = parseFloat (amount);\n        let request = {\n            'TradePairId': market['id'],\n            'Type': this.capitalize (side),\n            // 'Rate': this.priceToPrecision (symbol, price),\n            // 'Amount': this.amountToPrecision (symbol, amount),\n            'Rate': price,\n            'Amount': amount,\n        };\n        let response = await this.privatePostSubmitTrade (this.extend (request, params));\n        if (!response)\n            throw new ExchangeError (this.id + ' createOrder returned unknown error: ' + this.json (response));\n        let id = undefined;\n        let filled = 0.0;\n        if ('Data' in response) {\n            if ('OrderId' in response['Data']) {\n                if (response['Data']['OrderId']) {\n                    id = response['Data']['OrderId'].toString ();\n                }\n            }\n            if ('FilledOrders' in response['Data']) {\n                let filledOrders = response['Data']['FilledOrders'];\n                let filledOrdersLength = filledOrders.length;\n                if (filledOrdersLength) {\n                    filled = undefined;\n                }\n            }\n        }\n        let timestamp = this.milliseconds ();\n        let order = {\n            'id': id,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': 'open',\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': price,\n            'cost': price * amount,\n            'amount': amount,\n            'remaining': amount,\n            'filled': filled,\n            'fee': undefined,\n            // 'trades': this.parseTrades (order['trades'], market),\n        };\n        if (id)\n            this.orders[id] = order;\n        return this.extend ({ 'info': response }, order);\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = undefined;\n        try {\n            response = await this.privatePostCancelTrade (this.extend ({\n                'Type': 'Trade',\n                'OrderId': id,\n            }, params));\n            if (id in this.orders)\n                this.orders[id]['status'] = 'canceled';\n        } catch (e) {\n            if (this.last_json_response) {\n                let message = this.safeString (this.last_json_response, 'Error');\n                if (message) {\n                    if (message.indexOf ('does not exist') >= 0)\n                        throw new OrderNotFound (this.id + ' cancelOrder() error: ' + this.last_http_response);\n                }\n            }\n            throw e;\n        }\n        return response;\n    }\n\n    parseOrder (order, market = undefined) {\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else if ('Market' in order) {\n            let id = order['Market'];\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            }\n        }\n        let timestamp = this.parse8601 (order['TimeStamp']);\n        let amount = this.safeFloat (order, 'Amount');\n        let remaining = this.safeFloat (order, 'Remaining');\n        let filled = amount - remaining;\n        return {\n            'id': order['OrderId'].toString (),\n            'info': this.omit (order, 'status'),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': order['status'],\n            'symbol': symbol,\n            'type': 'limit',\n            'side': order['Type'].toLowerCase (),\n            'price': this.safeFloat (order, 'Rate'),\n            'cost': this.safeFloat (order, 'Total'),\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'fee': undefined,\n            // 'trades': this.parseTrades (order['trades'], market),\n        };\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrders requires a symbol param');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.privatePostGetOpenOrders ({\n            // 'Market': market['id'],\n            'TradePairId': market['id'], // Cryptopia identifier (not required if 'Market' supplied)\n            // 'Count': 100, // default = 100\n        }, params);\n        let orders = [];\n        for (let i = 0; i < response['Data'].length; i++) {\n            orders.push (this.extend (response['Data'][i], { 'status': 'open' }));\n        }\n        let openOrders = this.parseOrders (orders, market);\n        for (let j = 0; j < openOrders.length; j++) {\n            this.orders[openOrders[j]['id']] = openOrders[j];\n        }\n        let openOrdersIndexedById = this.indexBy (openOrders, 'id');\n        let cachedOrderIds = Object.keys (this.orders);\n        let result = [];\n        for (let k = 0; k < cachedOrderIds.length; k++) {\n            let id = cachedOrderIds[k];\n            if (id in openOrdersIndexedById) {\n                this.orders[id] = this.extend (this.orders[id], openOrdersIndexedById[id]);\n            } else {\n                let order = this.orders[id];\n                if (order['status'] === 'open') {\n                    this.orders[id] = this.extend (order, {\n                        'status': 'closed',\n                        'cost': order['amount'] * order['price'],\n                        'filled': order['amount'],\n                        'remaining': 0.0,\n                    });\n                }\n            }\n            let order = this.orders[id];\n            if (order['symbol'] === symbol)\n                result.push (order);\n        }\n        return this.filterBySinceLimit (result, since, limit);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        id = id.toString ();\n        let orders = await this.fetchOrders (symbol, undefined, undefined, params);\n        for (let i = 0; i < orders.length; i++) {\n            if (orders[i]['id'] === id)\n                return orders[i];\n        }\n        throw new OrderNotCached (this.id + ' order ' + id + ' not found in cached .orders, fetchOrder requires .orders (de)serialization implemented for this method to work properly');\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, params);\n        let result = [];\n        for (let i = 0; i < orders.length; i++) {\n            if (orders[i]['status'] === 'open')\n                result.push (orders[i]);\n        }\n        return result;\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, params);\n        let result = [];\n        for (let i = 0; i < orders.length; i++) {\n            if (orders[i]['status'] === 'closed')\n                result.push (orders[i]);\n        }\n        return result;\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        let currencyId = this.currencyId (currency);\n        let response = await this.privatePostGetDepositAddress (this.extend ({\n            'Currency': currencyId,\n        }, params));\n        let address = this.safeString (response['Data'], 'BaseAddress');\n        if (!address)\n            address = this.safeString (response['Data'], 'Address');\n        return {\n            'currency': currency,\n            'address': address,\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        let currencyId = this.currencyId (currency);\n        let request = {\n            'Currency': currencyId,\n            'Amount': amount,\n            'Address': address, // Address must exist in you AddressBook in security settings\n        };\n        if (tag)\n            request['PaymentId'] = tag;\n        let response = await this.privatePostSubmitWithdraw (this.extend (request, params));\n        return {\n            'info': response,\n            'id': response['Data'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            body = this.json (query);\n            let hash = this.hash (this.encode (body), 'md5', 'base64');\n            let secret = this.base64ToBinary (this.secret);\n            let uri = this.encodeURIComponent (url);\n            let lowercase = uri.toLowerCase ();\n            let payload = this.apiKey + method + lowercase + nonce + this.binaryToString (hash);\n            let signature = this.hmac (this.encode (payload), secret, 'sha256', 'base64');\n            let auth = 'amx ' + this.apiKey + ':' + this.binaryToString (signature) + ':' + nonce;\n            headers = {\n                'Content-Type': 'application/json',\n                'Authorization': auth,\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (response) {\n            if ('Success' in response)\n                if (response['Success']) {\n                    return response;\n                } else if ('Error' in response) {\n                    if (response['Error'] === 'Insufficient Funds.')\n                        throw new InsufficientFunds (this.id + ' ' + this.json (response));\n                }\n        }\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst liqui = require ('./liqui.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class dsx extends liqui {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'dsx',\n            'name': 'DSX',\n            'countries': 'UK',\n            'rateLimit': 1500,\n            'hasCORS': false,\n            'hasFetchOrder': true,\n            'hasFetchOrders': true,\n            'hasFetchOpenOrders': true,\n            'hasFetchClosedOrders': true,\n            'hasFetchTickers': true,\n            'hasFetchMyTrades': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27990275-1413158a-645a-11e7-931c-94717f7510e3.jpg',\n                'api': {\n                    'public': 'https://dsx.uk/mapi', // market data\n                    'private': 'https://dsx.uk/tapi', // trading\n                    'dwapi': 'https://dsx.uk/dwapi', // deposit/withdraw\n                },\n                'www': 'https://dsx.uk',\n                'doc': [\n                    'https://api.dsx.uk',\n                    'https://dsx.uk/api_docs/public',\n                    'https://dsx.uk/api_docs/private',\n                    '',\n                ],\n            },\n            'api': {\n                // market data (public)\n                'public': {\n                    'get': [\n                        'barsFromMoment/{id}/{period}/{start}', // empty reply :\\\n                        'depth/{pair}',\n                        'info',\n                        'lastBars/{id}/{period}/{amount}', // period is (m, h or d)\n                        'periodBars/{id}/{period}/{start}/{end}',\n                        'ticker/{pair}',\n                        'trades/{pair}',\n                    ],\n                },\n                // trading (private)\n                'private': {\n                    'post': [\n                        'getInfo',\n                        'TransHistory',\n                        'TradeHistory',\n                        'OrderHistory',\n                        'ActiveOrders',\n                        'Trade',\n                        'CancelOrder',\n                    ],\n                },\n                // deposit / withdraw (private)\n                'dwapi': {\n                    'post': [\n                        'getCryptoDepositAddress',\n                        'cryptoWithdraw',\n                        'fiatWithdraw',\n                        'getTransactionStatus',\n                        'getTransactions',\n                    ],\n                },\n            },\n        });\n    }\n\n    getBaseQuoteFromMarketId (id) {\n        let uppercase = id.toUpperCase ();\n        let base = uppercase.slice (0, 3);\n        let quote = uppercase.slice (3, 6);\n        base = this.commonCurrencyCode (base);\n        quote = this.commonCurrencyCode (quote);\n        return [ base, quote ];\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetInfo ();\n        let balances = response['return'];\n        let result = { 'info': balances };\n        let funds = balances['funds'];\n        let currencies = Object.keys (funds);\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let uppercase = currency.toUpperCase ();\n            uppercase = this.commonCurrencyCode (uppercase);\n            let account = {\n                'free': funds[currency],\n                'used': 0.0,\n                'total': balances['total'][currency],\n            };\n            account['used'] = account['total'] - account['free'];\n            result[uppercase] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['updated'] * 1000;\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'buy'),\n            'ask': this.safeFloat (ticker, 'sell'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': 1 / this.safeFloat (ticker, 'avg'),\n            'baseVolume': this.safeFloat (ticker, 'vol'),\n            'quoteVolume': this.safeFloat (ticker, 'vol_cur'),\n            'info': ticker,\n        };\n    }\n\n    getOrderIdKey () {\n        return 'orderId';\n    }\n\n    signBodyWithSecret (body) {\n        return this.decode (this.hmac (this.encode (body), this.encode (this.secret), 'sha512', 'base64'));\n    }\n\n    getVersionString () {\n        return ''; // they don't prepend version number to public URLs as other BTC-e clones do\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class exmo extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'exmo',\n            'name': 'EXMO',\n            'countries': [ 'ES', 'RU' ], // Spain, Russia\n            'rateLimit': 1000, // once every 350 ms ≈ 180 requests per minute ≈ 3 requests per second\n            'version': 'v1',\n            'hasCORS': false,\n            'hasFetchTickers': true,\n            'hasWithdraw': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766491-1b0ea956-5eda-11e7-9225-40d67b481b8d.jpg',\n                'api': 'https://api.exmo.com',\n                'www': 'https://exmo.me',\n                'doc': [\n                    'https://exmo.me/en/api_doc',\n                    'https://github.com/exmo-dev/exmo_api_lib/tree/master/nodejs',\n                ],\n                'fees': 'https://exmo.com/en/docs/fees',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'currency',\n                        'order_book',\n                        'pair_settings',\n                        'ticker',\n                        'trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'user_info',\n                        'order_create',\n                        'order_cancel',\n                        'user_open_orders',\n                        'user_trades',\n                        'user_cancelled_orders',\n                        'order_trades',\n                        'required_amount',\n                        'deposit_address',\n                        'withdraw_crypt',\n                        'withdraw_get_txid',\n                        'excode_create',\n                        'excode_load',\n                        'wallet_history',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.2 / 100,\n                    'taker': 0.2 / 100,\n                },\n                'funding': {\n                    'witdhraw': {\n                        'BTC': 0.001,\n                        'LTC': 0.01,\n                        'DOGE': 1,\n                        'DASH': 0.01,\n                        'ETH': 0.01,\n                        'WAVES': 0.001,\n                        'ZEC': 0.001,\n                        'USDT': 25,\n                        'XMR': 0.05,\n                        'XRP': 0.02,\n                        'KICK': 350,\n                        'ETC': 0.01,\n                        'BCH': 0.001,\n                    },\n                    'deposit': {\n                        'USDT': 15,\n                        'KICK': 50,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetPairSettings ();\n        let keys = Object.keys (markets);\n        let result = [];\n        for (let p = 0; p < keys.length; p++) {\n            let id = keys[p];\n            let market = markets[id];\n            let symbol = id.replace ('_', '/');\n            let [ base, quote ] = symbol.split ('/');\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'limits': {\n                    'amount': {\n                        'min': market['min_quantity'],\n                        'max': market['max_quantity'],\n                    },\n                    'price': {\n                        'min': market['min_price'],\n                        'max': market['max_price'],\n                    },\n                    'cost': {\n                        'min': market['min_amount'],\n                        'max': market['max_amount'],\n                    },\n                },\n                'precision': {\n                    'amount': 8,\n                    'price': 8,\n                },\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostUserInfo ();\n        let result = { 'info': response };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            if (currency in response['balances'])\n                account['free'] = parseFloat (response['balances'][currency]);\n            if (currency in response['reserved'])\n                account['used'] = parseFloat (response['reserved'][currency]);\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetOrderBook (this.extend ({\n            'pair': market['id'],\n        }, params));\n        let result = response[market['id']];\n        let orderbook = this.parseOrderBook (result, undefined, 'bid', 'ask');\n        return this.extend (orderbook, {\n            'bids': this.sortBy (orderbook['bids'], 0, true),\n            'asks': this.sortBy (orderbook['asks'], 0),\n        });\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['updated'] * 1000;\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy_price']),\n            'ask': parseFloat (ticker['sell_price']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last_trade']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': parseFloat (ticker['avg']),\n            'baseVolume': parseFloat (ticker['vol']),\n            'quoteVolume': parseFloat (ticker['vol_curr']),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetTicker (params);\n        let result = {};\n        let ids = Object.keys (response);\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = response[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetTicker (params);\n        let market = this.market (symbol);\n        return this.parseTicker (response[market['id']], market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['date'] * 1000;\n        return {\n            'id': trade['trade_id'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': undefined,\n            'type': undefined,\n            'side': trade['type'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['quantity']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTrades (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTrades (response[market['id']], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let prefix = '';\n        if (type === 'market')\n            prefix = 'market_';\n        if (typeof price === 'undefined')\n            price = 0;\n        let order = {\n            'pair': this.marketId (symbol),\n            'quantity': amount,\n            'price': price,\n            'type': prefix + side,\n        };\n        let response = await this.privatePostOrderCreate (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['order_id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostOrderCancel ({ 'order_id': id });\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let result = await this.privatePostWithdrawCrypt (this.extend ({\n            'amount': amount,\n            'currency': currency,\n            'address': address,\n        }, params));\n        return {\n            'info': result,\n            'id': result['task_id'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + path;\n        if (api === 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({ 'nonce': nonce }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Key': this.apiKey,\n                'Sign': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('result' in response) {\n            if (response['result'])\n                return response;\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class flowbtc extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'flowbtc',\n            'name': 'flowBTC',\n            'countries': 'BR', // Brazil\n            'version': 'v1',\n            'rateLimit': 1000,\n            'hasCORS': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28162465-cd815d4c-67cf-11e7-8e57-438bea0523a2.jpg',\n                'api': 'https://api.flowbtc.com:8400/ajax',\n                'www': 'https://trader.flowbtc.com',\n                'doc': 'http://www.flowbtc.com.br/api/',\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'uid': true,\n            },\n            'api': {\n                'public': {\n                    'post': [\n                        'GetTicker',\n                        'GetTrades',\n                        'GetTradesByDate',\n                        'GetOrderBook',\n                        'GetProductPairs',\n                        'GetProducts',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'CreateAccount',\n                        'GetUserInfo',\n                        'SetUserInfo',\n                        'GetAccountInfo',\n                        'GetAccountTrades',\n                        'GetDepositAddresses',\n                        'Withdraw',\n                        'CreateOrder',\n                        'ModifyOrder',\n                        'CancelOrder',\n                        'CancelAllOrders',\n                        'GetAccountOpenOrders',\n                        'GetOrderFee',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicPostGetProductPairs ();\n        let markets = response['productPairs'];\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['name'];\n            let base = market['product1Label'];\n            let quote = market['product2Label'];\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetAccountInfo ();\n        let balances = response['currencies'];\n        let result = { 'info': response };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['name'];\n            let account = {\n                'free': balance['balance'],\n                'used': balance['hold'],\n                'total': 0.0,\n            };\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orderbook = await this.publicPostGetOrderBook (this.extend ({\n            'productPair': market['id'],\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'px', 'qty');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicPostGetTicker (this.extend ({\n            'productPair': market['id'],\n        }, params));\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume24hr']),\n            'quoteVolume': parseFloat (ticker['volume24hrProduct2']),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['unixtime'] * 1000;\n        let side = (trade['incomingOrderSide'] == 0) ? 'buy' : 'sell';\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': trade['tid'].toString (),\n            'order': undefined,\n            'type': undefined,\n            'side': side,\n            'price': trade['px'],\n            'amount': trade['qty'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicPostGetTrades (this.extend ({\n            'ins': market['id'],\n            'startIndex': -1,\n        }, params));\n        return this.parseTrades (response['trades'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let orderType = (type == 'market') ? 1 : 0;\n        let order = {\n            'ins': this.marketId (symbol),\n            'side': side,\n            'orderType': orderType,\n            'qty': amount,\n            'px': price,\n        };\n        let response = await this.privatePostCreateOrder (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['serverOrderId'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        if ('ins' in params) {\n            return await this.privatePostCancelOrder (this.extend ({\n                'serverOrderId': id,\n            }, params));\n        }\n        throw new ExchangeError (this.id + ' requires `ins` symbol parameter for cancelling an order');\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + path;\n        if (api == 'public') {\n            if (Object.keys (params).length) {\n                body = this.json (params);\n            }\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let auth = nonce.toString () + this.uid + this.apiKey;\n            let signature = this.hmac (this.encode (auth), this.encode (this.secret));\n            body = this.json (this.extend ({\n                'apiKey': this.apiKey,\n                'apiNonce': nonce,\n                'apiSig': signature.toUpperCase (),\n            }, params));\n            headers = {\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('isAccepted' in response)\n            if (response['isAccepted'])\n                return response;\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class foxbit extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'foxbit',\n            'name': 'FoxBit',\n            'countries': 'BR',\n            'hasCORS': false,\n            'rateLimit': 1000,\n            'version': 'v1',\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27991413-11b40d42-647f-11e7-91ee-78ced874dd09.jpg',\n                'api': {\n                    'public': 'https://api.blinktrade.com/api',\n                    'private': 'https://api.blinktrade.com/tapi',\n                },\n                'www': 'https://foxbit.exchange',\n                'doc': 'https://blinktrade.com/docs',\n            },\n            'comment': 'Blinktrade API',\n            'api': {\n                'public': {\n                    'get': [\n                        '{currency}/ticker',    // ?crypto_currency=BTC\n                        '{currency}/orderbook', // ?crypto_currency=BTC\n                        '{currency}/trades',    // ?crypto_currency=BTC&since=<TIMESTAMP>&limit=<NUMBER>\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'D',   // order\n                        'F',   // cancel order\n                        'U2',  // balance\n                        'U4',  // my orders\n                        'U6',  // withdraw\n                        'U18', // deposit\n                        'U24', // confirm withdrawal\n                        'U26', // list withdrawals\n                        'U30', // list deposits\n                        'U34', // ledger\n                        'U70', // cancel withdrawal\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/VEF': { 'id': 'BTCVEF', 'symbol': 'BTC/VEF', 'base': 'BTC', 'quote': 'VEF', 'brokerId': 1, 'broker': 'SurBitcoin' },\n                'BTC/VND': { 'id': 'BTCVND', 'symbol': 'BTC/VND', 'base': 'BTC', 'quote': 'VND', 'brokerId': 3, 'broker': 'VBTC' },\n                'BTC/BRL': { 'id': 'BTCBRL', 'symbol': 'BTC/BRL', 'base': 'BTC', 'quote': 'BRL', 'brokerId': 4, 'broker': 'FoxBit' },\n                'BTC/PKR': { 'id': 'BTCPKR', 'symbol': 'BTC/PKR', 'base': 'BTC', 'quote': 'PKR', 'brokerId': 8, 'broker': 'UrduBit' },\n                'BTC/CLP': { 'id': 'BTCCLP', 'symbol': 'BTC/CLP', 'base': 'BTC', 'quote': 'CLP', 'brokerId': 9, 'broker': 'ChileBit' },\n            },\n        });\n    }\n\n    fetchBalance (params = {}) {\n        // todo parse balance\n        return this.privatePostU2 ({\n            'BalanceReqID': this.nonce (),\n        });\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let market = this.market (symbol);\n        let orderbook = await this.publicGetCurrencyOrderbook (this.extend ({\n            'currency': market['quote'],\n            'crypto_currency': market['base'],\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let market = this.market (symbol);\n        let ticker = await this.publicGetCurrencyTicker (this.extend ({\n            'currency': market['quote'],\n            'crypto_currency': market['base'],\n        }, params));\n        let timestamp = this.milliseconds ();\n        let lowercaseQuote = market['quote'].toLowerCase ();\n        let quoteVolume = 'vol_' + lowercaseQuote;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy']),\n            'ask': parseFloat (ticker['sell']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['vol']),\n            'quoteVolume': parseFloat (ticker[quoteVolume]),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['date'] * 1000;\n        return {\n            'id': trade['tid'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['side'],\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetCurrencyTrades (this.extend ({\n            'currency': market['quote'],\n            'crypto_currency': market['base'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type == 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let market = this.market (symbol);\n        let orderSide = (side == 'buy') ? '1' : '2';\n        let order = {\n            'ClOrdID': this.nonce (),\n            'Symbol': market['id'],\n            'Side': orderSide,\n            'OrdType': '2',\n            'Price': price,\n            'OrderQty': amount,\n            'BrokerID': market['brokerId'],\n        };\n        let response = await this.privatePostD (this.extend (order, params));\n        let indexed = this.indexBy (response['Responses'], 'MsgType');\n        let execution = indexed['8'];\n        return {\n            'info': response,\n            'id': execution['OrderID'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostF (this.extend ({\n            'ClOrdID': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api] + '/' + this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let request = this.extend ({ 'MsgType': path }, query);\n            body = this.json (request);\n            headers = {\n                'APIKey': this.apiKey,\n                'Nonce': nonce,\n                'Signature': this.hmac (this.encode (nonce), this.encode (this.secret)),\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('Status' in response)\n            if (response['Status'] != 200)\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class fybse extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'fybse',\n            'name': 'FYB-SE',\n            'countries': 'SE', // Sweden\n            'hasCORS': false,\n            'rateLimit': 1500,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766512-31019772-5edb-11e7-8241-2e675e6797f1.jpg',\n                'api': 'https://www.fybse.se/api/SEK',\n                'www': 'https://www.fybse.se',\n                'doc': 'http://docs.fyb.apiary.io',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'ticker',\n                        'tickerdetailed',\n                        'orderbook',\n                        'trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'test',\n                        'getaccinfo',\n                        'getpendingorders',\n                        'getorderhistory',\n                        'cancelpendingorder',\n                        'placeorder',\n                        'withdraw',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/SEK': { 'id': 'SEK', 'symbol': 'BTC/SEK', 'base': 'BTC', 'quote': 'SEK' },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let balance = await this.privatePostGetaccinfo ();\n        let btc = parseFloat (balance['btcBal']);\n        let symbol = this.symbols[0];\n        let quote = this.markets[symbol]['quote'];\n        let lowercase = quote.toLowerCase () + 'Bal';\n        let fiat = parseFloat (balance[lowercase]);\n        let crypto = {\n            'free': btc,\n            'used': 0.0,\n            'total': btc,\n        };\n        let result = { 'BTC': crypto };\n        result[quote] = {\n            'free': fiat,\n            'used': 0.0,\n            'total': fiat,\n        };\n        result['info'] = balance;\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetOrderbook (params);\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetTickerdetailed (params);\n        let timestamp = this.milliseconds ();\n        let last = undefined;\n        let volume = undefined;\n        if ('last' in ticker)\n            last = parseFloat (ticker['last']);\n        if ('vol' in ticker)\n            volume = parseFloat (ticker['vol']);\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': last,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': volume,\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        return {\n            'info': trade,\n            'id': trade['tid'].toString (),\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetTrades (params);\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let response = await this.privatePostPlaceorder (this.extend ({\n            'qty': amount,\n            'price': price,\n            'type': side[0].toUpperCase (),\n        }, params));\n        return {\n            'info': response,\n            'id': response['pending_oid'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelpendingorder ({ 'orderNo': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + path;\n        if (api == 'public') {\n            url += '.json';\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({ 'timestamp': nonce }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'key': this.apiKey,\n                'sig': this.hmac (this.encode (body), this.encode (this.secret), 'sha1'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (api == 'private')\n            if ('error' in response)\n                if (response['error'])\n                    throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst fybse = require ('./fybse.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class fybsg extends fybse {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'fybsg',\n            'name': 'FYB-SG',\n            'countries': 'SG', // Singapore\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766513-3364d56a-5edb-11e7-9e6b-d5898bb89c81.jpg',\n                'api': 'https://www.fybsg.com/api/SGD',\n                'www': 'https://www.fybsg.com',\n                'doc': 'http://docs.fyb.apiary.io',\n            },\n            'markets': {\n                'BTC/SGD': { 'id': 'SGD', 'symbol': 'BTC/SGD', 'base': 'BTC', 'quote': 'SGD' },\n            },\n        });\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, AuthenticationError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class gatecoin extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'gatecoin',\n            'name': 'Gatecoin',\n            'rateLimit': 2000,\n            'countries': 'HK', // Hong Kong\n            'comment': 'a regulated/licensed exchange',\n            'hasCORS': false,\n            'hasFetchTickers': true,\n            'hasFetchOHLCV': true,\n            'timeframes': {\n                '1m': '1m',\n                '15m': '15m',\n                '1h': '1h',\n                '6h': '6h',\n                '1d': '24h',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28646817-508457f2-726c-11e7-9eeb-3528d2413a58.jpg',\n                'api': 'https://api.gatecoin.com',\n                'www': 'https://gatecoin.com',\n                'doc': [\n                    'https://gatecoin.com/api',\n                    'https://github.com/Gatecoin/RESTful-API-Implementation',\n                    'https://api.gatecoin.com/swagger-ui/index.html',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'Public/ExchangeRate', // Get the exchange rates\n                        'Public/LiveTicker', // Get live ticker for all currency\n                        'Public/LiveTicker/{CurrencyPair}', // Get live ticker by currency\n                        'Public/LiveTickers', // Get live ticker for all currency\n                        'Public/MarketDepth/{CurrencyPair}', // Gets prices and market depth for the currency pair.\n                        'Public/NetworkStatistics/{DigiCurrency}', // Get the network status of a specific digital currency\n                        'Public/StatisticHistory/{DigiCurrency}/{Typeofdata}', // Get the historical data of a specific digital currency\n                        'Public/TickerHistory/{CurrencyPair}/{Timeframe}', // Get ticker history\n                        'Public/Transactions/{CurrencyPair}', // Gets recent transactions\n                        'Public/TransactionsHistory/{CurrencyPair}', // Gets all transactions\n                        'Reference/BusinessNatureList', // Get the business nature list.\n                        'Reference/Countries', // Get the country list.\n                        'Reference/Currencies', // Get the currency list.\n                        'Reference/CurrencyPairs', // Get the currency pair list.\n                        'Reference/CurrentStatusList', // Get the current status list.\n                        'Reference/IdentydocumentTypes', // Get the different types of identity documents possible.\n                        'Reference/IncomeRangeList', // Get the income range list.\n                        'Reference/IncomeSourceList', // Get the income source list.\n                        'Reference/VerificationLevelList', // Get the verif level list.\n                        'Stream/PublicChannel', // Get the public pubnub channel list\n                    ],\n                    'post': [\n                        'Export/Transactions', // Request a export of all trades from based on currencypair, start date and end date\n                        'Ping', // Post a string, then get it back.\n                        'Public/Unsubscribe/{EmailCode}', // Lets the user unsubscribe from emails\n                        'RegisterUser', // Initial trader registration.\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'Account/CorporateData', // Get corporate account data\n                        'Account/DocumentAddress', // Check if residence proof uploaded\n                        'Account/DocumentCorporation', // Check if registered document uploaded\n                        'Account/DocumentID', // Check if ID document copy uploaded\n                        'Account/DocumentInformation', // Get Step3 Data\n                        'Account/Email', // Get user email\n                        'Account/FeeRate', // Get fee rate of logged in user\n                        'Account/Level', // Get verif level of logged in user\n                        'Account/PersonalInformation', // Get Step1 Data\n                        'Account/Phone', // Get user phone number\n                        'Account/Profile', // Get trader profile\n                        'Account/Questionnaire', // Fill the questionnaire\n                        'Account/Referral', // Get referral information\n                        'Account/ReferralCode', // Get the referral code of the logged in user\n                        'Account/ReferralNames', // Get names of referred traders\n                        'Account/ReferralReward', // Get referral reward information\n                        'Account/ReferredCode', // Get referral code\n                        'Account/ResidentInformation', // Get Step2 Data\n                        'Account/SecuritySettings', // Get verif details of logged in user\n                        'Account/User', // Get all user info\n                        'APIKey/APIKey', // Get API Key for logged in user\n                        'Auth/ConnectionHistory', // Gets connection history of logged in user\n                        'Balance/Balances', // Gets the available balance for each currency for the logged in account.\n                        'Balance/Balances/{Currency}', // Gets the available balance for s currency for the logged in account.\n                        'Balance/Deposits', // Get all account deposits, including wire and digital currency, of the logged in user\n                        'Balance/Withdrawals', // Get all account withdrawals, including wire and digital currency, of the logged in user\n                        'Bank/Accounts/{Currency}/{Location}', // Get internal bank account for deposit\n                        'Bank/Transactions', // Get all account transactions of the logged in user\n                        'Bank/UserAccounts', // Gets all the bank accounts related to the logged in user.\n                        'Bank/UserAccounts/{Currency}', // Gets all the bank accounts related to the logged in user.\n                        'ElectronicWallet/DepositWallets', // Gets all crypto currency addresses related deposits to the logged in user.\n                        'ElectronicWallet/DepositWallets/{DigiCurrency}', // Gets all crypto currency addresses related deposits to the logged in user by currency.\n                        'ElectronicWallet/Transactions', // Get all digital currency transactions of the logged in user\n                        'ElectronicWallet/Transactions/{DigiCurrency}', // Get all digital currency transactions of the logged in user\n                        'ElectronicWallet/UserWallets', // Gets all external digital currency addresses related to the logged in user.\n                        'ElectronicWallet/UserWallets/{DigiCurrency}', // Gets all external digital currency addresses related to the logged in user by currency.\n                        'Info/ReferenceCurrency', // Get user's reference currency\n                        'Info/ReferenceLanguage', // Get user's reference language\n                        'Notification/Messages', // Get from oldest unread + 3 read message to newest messages\n                        'Trade/Orders', // Gets open orders for the logged in trader.\n                        'Trade/Orders/{OrderID}', // Gets an order for the logged in trader.\n                        'Trade/StopOrders', // Gets all stop orders for the logged in trader. Max 1000 record.\n                        'Trade/StopOrdersHistory', // Gets all stop orders for the logged in trader. Max 1000 record.\n                        'Trade/Trades', // Gets all transactions of logged in user\n                        'Trade/UserTrades', // Gets all transactions of logged in user\n                    ],\n                    'post': [\n                        'Account/DocumentAddress', // Upload address proof document\n                        'Account/DocumentCorporation', // Upload registered document document\n                        'Account/DocumentID', // Upload ID document copy\n                        'Account/Email/RequestVerify', // Request for verification email\n                        'Account/Email/Verify', // Verification email\n                        'Account/GoogleAuth', // Enable google auth\n                        'Account/Level', // Request verif level of logged in user\n                        'Account/Questionnaire', // Fill the questionnaire\n                        'Account/Referral', // Post a referral email\n                        'APIKey/APIKey', // Create a new API key for logged in user\n                        'Auth/ChangePassword', // Change password.\n                        'Auth/ForgotPassword', // Request reset password\n                        'Auth/ForgotUserID', // Request user id\n                        'Auth/Login', // Trader session log in.\n                        'Auth/Logout', // Logout from the current session.\n                        'Auth/LogoutOtherSessions', // Logout other sessions.\n                        'Auth/ResetPassword', // Reset password\n                        'Bank/Transactions', // Request a transfer from the traders account of the logged in user. This is only available for bank account\n                        'Bank/UserAccounts', // Add an account the logged in user\n                        'ElectronicWallet/DepositWallets/{DigiCurrency}', // Add an digital currency addresses to the logged in user.\n                        'ElectronicWallet/Transactions/Deposits/{DigiCurrency}', // Get all internal digital currency transactions of the logged in user\n                        'ElectronicWallet/Transactions/Withdrawals/{DigiCurrency}', // Get all external digital currency transactions of the logged in user\n                        'ElectronicWallet/UserWallets/{DigiCurrency}', // Add an external digital currency addresses to the logged in user.\n                        'ElectronicWallet/Withdrawals/{DigiCurrency}', // Request a transfer from the traders account to an external address. This is only available for crypto currencies.\n                        'Notification/Messages', // Mark all as read\n                        'Notification/Messages/{ID}', // Mark as read\n                        'Trade/Orders', // Place an order at the exchange.\n                        'Trade/StopOrders', // Place a stop order at the exchange.\n                    ],\n                    'put': [\n                        'Account/CorporateData', // Update user company data for corporate account\n                        'Account/DocumentID', // Update ID document meta data\n                        'Account/DocumentInformation', // Update Step3 Data\n                        'Account/Email', // Update user email\n                        'Account/PersonalInformation', // Update Step1 Data\n                        'Account/Phone', // Update user phone number\n                        'Account/Questionnaire', // update the questionnaire\n                        'Account/ReferredCode', // Update referral code\n                        'Account/ResidentInformation', // Update Step2 Data\n                        'Account/SecuritySettings', // Update verif details of logged in user\n                        'Account/User', // Update all user info\n                        'Bank/UserAccounts', // Update the label of existing user bank accounnt\n                        'ElectronicWallet/DepositWallets/{DigiCurrency}/{AddressName}', // Update the name of an address\n                        'ElectronicWallet/UserWallets/{DigiCurrency}', // Update the name of an external address\n                        'Info/ReferenceCurrency', // User's reference currency\n                        'Info/ReferenceLanguage', // Update user's reference language\n                    ],\n                    'delete': [\n                        'APIKey/APIKey/{PublicKey}', // Remove an API key\n                        'Bank/Transactions/{RequestID}', // Delete pending account withdraw of the logged in user\n                        'Bank/UserAccounts/{Currency}/{Label}', // Delete an account of the logged in user\n                        'ElectronicWallet/DepositWallets/{DigiCurrency}/{AddressName}', // Delete an digital currency addresses related to the logged in user.\n                        'ElectronicWallet/UserWallets/{DigiCurrency}/{AddressName}', // Delete an external digital currency addresses related to the logged in user.\n                        'Trade/Orders', // Cancels all existing order\n                        'Trade/Orders/{OrderID}', // Cancels an existing order\n                        'Trade/StopOrders', // Cancels all existing stop orders\n                        'Trade/StopOrders/{ID}', // Cancels an existing stop order\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.0025,\n                    'taker': 0.0035,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicGetPublicLiveTickers ();\n        let markets = response['tickers'];\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['currencyPair'];\n            let base = id.slice (0, 3);\n            let quote = id.slice (3, 6);\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetBalanceBalances ();\n        let balances = response['balances'];\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let account = {\n                'free': balance['availableBalance'],\n                'used': this.sum (\n                    balance['pendingIncoming'],\n                    balance['pendingOutgoing'],\n                    balance['openOrder']),\n                'total': balance['balance'],\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orderbook = await this.publicGetPublicMarketDepthCurrencyPair (this.extend ({\n            'CurrencyPair': market['id'],\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'volume');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = parseInt (ticker['createDateTime']) * 1000;\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let baseVolume = parseFloat (ticker['volume']);\n        let vwap = parseFloat (ticker['vwap']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': vwap,\n            'open': parseFloat (ticker['open']),\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetPublicLiveTickers (params);\n        let tickers = response['tickers'];\n        let result = {};\n        for (let t = 0; t < tickers.length; t++) {\n            let ticker = tickers[t];\n            let id = ticker['currencyPair'];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetPublicLiveTickerCurrencyPair (this.extend ({\n            'CurrencyPair': market['id'],\n        }, params));\n        let ticker = response['ticker'];\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let side = undefined;\n        let order = undefined;\n        if ('way' in trade) {\n            side = (trade['way'] == 'bid') ? 'buy' : 'sell';\n            let orderId = trade['way'] + 'OrderId';\n            order = trade[orderId];\n        }\n        let timestamp = parseInt (trade['transactionTime']) * 1000;\n        if (!market)\n            market = this.markets_by_id[trade['currencyPair']];\n        return {\n            'info': trade,\n            'id': trade['transactionId'].toString (),\n            'order': order,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': side,\n            'price': trade['price'],\n            'amount': trade['quantity'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetPublicTransactionsCurrencyPair (this.extend ({\n            'CurrencyPair': market['id'],\n        }, params));\n        return this.parseTrades (response['transactions'], market, since, limit);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            parseInt (ohlcv['createDateTime']) * 1000,\n            ohlcv['open'],\n            ohlcv['high'],\n            ohlcv['low'],\n            undefined,\n            ohlcv['volume'],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'CurrencyPair': market['id'],\n            'Timeframe': this.timeframes[timeframe],\n        };\n        if (limit)\n            request['Count'] = limit;\n        request = this.extend (request, params);\n        let response = await this.publicGetPublicTickerHistoryCurrencyPairTimeframe (request);\n        return this.parseOHLCVs (response['tickers'], market, timeframe, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'Code': this.marketId (symbol),\n            'Way': (side == 'buy') ? 'Bid' : 'Ask',\n            'Amount': amount,\n        };\n        if (type == 'limit')\n            order['Price'] = price;\n        if (this.twofa) {\n            if ('ValidationCode' in params)\n                order['ValidationCode'] = params['ValidationCode'];\n            else\n                throw new AuthenticationError (this.id + ' two-factor authentication requires a missing ValidationCode parameter');\n        }\n        let response = await this.privatePostTradeOrders (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['clOrderId'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privateDeleteTradeOrdersOrderID ({ 'OrderID': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let nonceString = nonce.toString ();\n            let contentType = (method == 'GET') ? '' : 'application/json';\n            let auth = method + url + contentType + nonceString;\n            auth = auth.toLowerCase ();\n            let signature = this.hmac (this.encode (auth), this.encode (this.secret), 'sha256', 'base64');\n            headers = {\n                'API_PUBLIC_KEY': this.apiKey,\n                'API_REQUEST_SIGNATURE': this.decode (signature),\n                'API_REQUEST_DATE': nonceString,\n            };\n            if (method != 'GET') {\n                headers['Content-Type'] = contentType;\n                body = this.json (this.extend ({ 'nonce': nonce }, params));\n            }\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('responseStatus' in response)\n            if ('message' in response['responseStatus'])\n                if (response['responseStatus']['message'] == 'OK')\n                    return response;\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst bter = require ('./bter.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class gateio extends bter {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'gateio',\n            'name': 'Gate.io',\n            'countries': 'CN',\n            'rateLimit': 1000,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/31784029-0313c702-b509-11e7-9ccc-bc0da6a0e435.jpg',\n                'api': {\n                    'public': 'https://data.gate.io/api',\n                    'private': 'https://data.gate.io/api',\n                },\n                'www': 'https://gate.io/',\n                'doc': 'https://gate.io/api2',\n                'fees': 'https://gate.io/fee',\n            },\n        });\n    }\n\n    parseTrade (trade, market) {\n        // exchange reports local time (UTC+8)\n        let timestamp = this.parse8601 (trade['date']) - 8 * 60 * 60 * 1000;\n        return {\n            'id': trade['tradeID'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['type'],\n            'price': trade['rate'],\n            'amount': this.safeFloat (trade, 'amount'),\n        };\n    }\n}\n","'use strict';\n\n// ----------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InvalidOrder, AuthenticationError, NotSupported } = require ('./base/errors');\n\n// ----------------------------------------------------------------------------\n\nmodule.exports = class gdax extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'gdax',\n            'name': 'GDAX',\n            'countries': 'US',\n            'rateLimit': 1000,\n            'userAgent': this.userAgents['chrome'],\n            // obsolete metainfo interface\n            'hasCORS': true,\n            'hasFetchOHLCV': true,\n            'hasDeposit': true,\n            'hasWithdraw': true,\n            'hasFetchOrder': true,\n            'hasFetchOrders': true,\n            'hasFetchOpenOrders': true,\n            'hasFetchClosedOrders': true,\n            // new metainfo interface\n            'has': {\n                'CORS': true,\n                'fetchOHLCV': true,\n                'deposit': true,\n                'withdraw': true,\n                'fetchOrder': true,\n                'fetchOrders': true,\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': true,\n            },\n            'timeframes': {\n                '1m': 60,\n                '5m': 300,\n                '15m': 900,\n                '30m': 1800,\n                '1h': 3600,\n                '2h': 7200,\n                '4h': 14400,\n                '12h': 43200,\n                '1d': 86400,\n                '1w': 604800,\n                '1M': 2592000,\n                '1y': 31536000,\n            },\n            'urls': {\n                'test': 'https://api-public.sandbox.gdax.com',\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766527-b1be41c6-5edb-11e7-95f6-5b496c469e2c.jpg',\n                'api': 'https://api.gdax.com',\n                'www': 'https://www.gdax.com',\n                'doc': 'https://docs.gdax.com',\n                'fees': [\n                    'https://www.gdax.com/fees',\n                    'https://support.gdax.com/customer/en/portal/topics/939402-depositing-and-withdrawing-funds/articles',\n                ],\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'password': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'currencies',\n                        'products',\n                        'products/{id}/book',\n                        'products/{id}/candles',\n                        'products/{id}/stats',\n                        'products/{id}/ticker',\n                        'products/{id}/trades',\n                        'time',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'accounts',\n                        'accounts/{id}',\n                        'accounts/{id}/holds',\n                        'accounts/{id}/ledger',\n                        'coinbase-accounts',\n                        'fills',\n                        'funding',\n                        'orders',\n                        'orders/{id}',\n                        'payment-methods',\n                        'position',\n                        'reports/{id}',\n                        'users/self/trailing-volume',\n                    ],\n                    'post': [\n                        'deposits/coinbase-account',\n                        'deposits/payment-method',\n                        'funding/repay',\n                        'orders',\n                        'position/close',\n                        'profiles/margin-transfer',\n                        'reports',\n                        'withdrawals/coinbase',\n                        'withdrawals/crypto',\n                        'withdrawals/payment-method',\n                    ],\n                    'delete': [\n                        'orders',\n                        'orders/{id}',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': true, // complicated tier system per coin\n                    'percentage': true,\n                    'maker': 0.0,\n                    'taker': 0.30 / 100, // worst-case scenario: https://www.gdax.com/fees/BTC-USD\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BCH': 0,\n                        'BTC': 0,\n                        'LTC': 0,\n                        'ETH': 0,\n                        'EUR': 0.15,\n                        'USD': 25,\n                    },\n                    'deposit': {\n                        'BCH': 0,\n                        'BTC': 0,\n                        'LTC': 0,\n                        'ETH': 0,\n                        'EUR': 0.15,\n                        'USD': 10,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetProducts ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['id'];\n            let base = market['base_currency'];\n            let quote = market['quote_currency'];\n            let symbol = base + '/' + quote;\n            let amountLimits = {\n                'min': market['base_min_size'],\n                'max': market['base_max_size'],\n            };\n            let priceLimits = {\n                'min': market['quote_increment'],\n                'max': undefined,\n            };\n            let costLimits = {\n                'min': priceLimits['min'],\n                'max': undefined,\n            };\n            let limits = {\n                'amount': amountLimits,\n                'price': priceLimits,\n                'cost': costLimits,\n            };\n            let precision = {\n                'amount': -Math.log10 (parseFloat (amountLimits['min'])),\n                'price': -Math.log10 (parseFloat (priceLimits['min'])),\n            };\n            let taker = this.fees['trading']['taker'];\n            if ((base === 'ETH') || (base === 'LTC')) {\n                taker = 0.003;\n            }\n            let active = market['status'] === 'online';\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n                'precision': precision,\n                'limits': limits,\n                'taker': taker,\n                'active': active,\n            }));\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privateGetAccounts ();\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let account = {\n                'free': parseFloat (balance['available']),\n                'used': parseFloat (balance['hold']),\n                'total': parseFloat (balance['balance']),\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetProductsIdBook (this.extend ({\n            'id': this.marketId (symbol),\n            'level': 2, // 1 best bidask, 2 aggregated, 3 full\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = this.extend ({\n            'id': market['id'],\n        }, params);\n        let ticker = await this.publicGetProductsIdTicker (request);\n        let timestamp = this.parse8601 (ticker['time']);\n        let bid = undefined;\n        let ask = undefined;\n        if ('bid' in ticker)\n            bid = parseFloat (ticker['bid']);\n        if ('ask' in ticker)\n            ask = parseFloat (ticker['ask']);\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': bid,\n            'ask': ask,\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'price'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = this.parse8601 (trade['time']);\n        let side = (trade['side'] === 'buy') ? 'sell' : 'buy';\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let fee = undefined;\n        if ('fill_fees' in trade) {\n            fee = {\n                'cost': parseFloat (trade['fill_fees']),\n                'currency': market['quote'],\n            };\n        }\n        return {\n            'id': trade['trade_id'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': undefined,\n            'side': side,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['size']),\n            'fee': fee,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetProductsIdTrades (this.extend ({\n            'id': market['id'], // fixes issue #2\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv[0] * 1000,\n            ohlcv[3],\n            ohlcv[2],\n            ohlcv[1],\n            ohlcv[4],\n            ohlcv[5],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let granularity = this.timeframes[timeframe];\n        let request = {\n            'id': market['id'],\n            'granularity': granularity,\n        };\n        if (since) {\n            request['start'] = this.YmdHMS (since);\n            if (!limit)\n                limit = 200; // max = 200\n            request['end'] = this.YmdHMS (this.sum (limit * granularity * 1000, since));\n        }\n        let response = await this.publicGetProductsIdCandles (this.extend (request, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    async fetchTime () {\n        let response = this.publicGetTime ();\n        return this.parse8601 (response['iso']);\n    }\n\n    parseOrderStatus (status) {\n        let statuses = {\n            'pending': 'open',\n            'active': 'open',\n            'open': 'open',\n            'done': 'closed',\n            'canceled': 'canceled',\n        };\n        return this.safeString (statuses, status, status);\n    }\n\n    parseOrder (order, market = undefined) {\n        let timestamp = this.parse8601 (order['created_at']);\n        let symbol = undefined;\n        if (!market) {\n            if (order['product_id'] in this.markets_by_id)\n                market = this.markets_by_id[order['product_id']];\n        }\n        let status = this.parseOrderStatus (order['status']);\n        let price = this.safeFloat (order, 'price');\n        let amount = this.safeFloat (order, 'size');\n        let filled = this.safeFloat (order, 'filled_size');\n        let remaining = amount - filled;\n        let cost = this.safeFloat (order, 'executed_value');\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'id': order['id'],\n            'info': order,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': status,\n            'symbol': symbol,\n            'type': order['type'],\n            'side': order['side'],\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'fee': undefined,\n        };\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetOrdersId (this.extend ({\n            'id': id,\n        }, params));\n        return this.parseOrder (response);\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            'status': 'all',\n        };\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['product_id'] = market['id'];\n        }\n        let response = await this.privateGetOrders (this.extend (request, params));\n        return this.parseOrders (response, market, since, limit);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['product_id'] = market['id'];\n        }\n        let response = await this.privateGetOrders (this.extend (request, params));\n        return this.parseOrders (response, market, since, limit);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            'status': 'done',\n        };\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['product_id'] = market['id'];\n        }\n        let response = await this.privateGetOrders (this.extend (request, params));\n        return this.parseOrders (response, market, since, limit);\n    }\n\n    async createOrder (market, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        // let oid = this.nonce ().toString ();\n        let order = {\n            'product_id': this.marketId (market),\n            'side': side,\n            'size': amount,\n            'type': type,\n        };\n        if (type === 'limit')\n            order['price'] = price;\n        let response = await this.privatePostOrders (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privateDeleteOrdersId ({ 'id': id });\n    }\n\n    async getPaymentMethods () {\n        let response = await this.privateGetPaymentMethods ();\n        return response;\n    }\n\n    async deposit (currency, amount, address, params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            'currency': currency,\n            'amount': amount,\n        };\n        let method = 'privatePostDeposits';\n        if ('payment_method_id' in params) {\n            // deposit from a payment_method, like a bank account\n            method += 'PaymentMethod';\n        } else if ('coinbase_account_id' in params) {\n            // deposit into GDAX account from a Coinbase account\n            method += 'CoinbaseAccount';\n        } else {\n            // deposit methodotherwise we did not receive a supported deposit location\n            // relevant docs link for the Googlers\n            // https://docs.gdax.com/#deposits\n            throw new NotSupported (this.id + ' deposit() requires one of `coinbase_account_id` or `payment_method_id` extra params');\n        }\n        let response = await this[method] (this.extend (request, params));\n        if (!response)\n            throw new ExchangeError (this.id + ' deposit() error: ' + this.json (response));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            'currency': currency,\n            'amount': amount,\n        };\n        let method = 'privatePostWithdrawals';\n        if ('payment_method_id' in params) {\n            method += 'PaymentMethod';\n        } else if ('coinbase_account_id' in params) {\n            method += 'CoinbaseAccount';\n        } else {\n            method += 'Crypto';\n            request['crypto_address'] = address;\n        }\n        let response = await this[method] (this.extend (request, params));\n        if (!response)\n            throw new ExchangeError (this.id + ' withdraw() error: ' + this.json (response));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let request = '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (method === 'GET') {\n            if (Object.keys (query).length)\n                request += '?' + this.urlencode (query);\n        }\n        let url = this.urls['api'] + request;\n        if (api === 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let payload = '';\n            if (method !== 'GET') {\n                if (Object.keys (query).length) {\n                    body = this.json (query);\n                    payload = body;\n                }\n            }\n            // let payload = (body) ? body : '';\n            let what = nonce + method + request + payload;\n            let secret = this.base64ToBinary (this.secret);\n            let signature = this.hmac (this.encode (what), secret, 'sha256', 'base64');\n            headers = {\n                'CB-ACCESS-KEY': this.apiKey,\n                'CB-ACCESS-SIGN': this.decode (signature),\n                'CB-ACCESS-TIMESTAMP': nonce,\n                'CB-ACCESS-PASSPHRASE': this.password,\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code === 400) {\n            if (body[0] === '{') {\n                let response = JSON.parse (body);\n                let message = response['message'];\n                if (message.indexOf ('price too small') >= 0) {\n                    throw new InvalidOrder (this.id + ' ' + message);\n                } else if (message.indexOf ('price too precise') >= 0) {\n                    throw new InvalidOrder (this.id + ' ' + message);\n                } else if (message === 'Invalid API Key') {\n                    throw new AuthenticationError (this.id + ' ' + message);\n                }\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n            }\n            throw new ExchangeError (this.id + ' ' + body);\n        }\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('message' in response) {\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class gemini extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'gemini',\n            'name': 'Gemini',\n            'countries': 'US',\n            'rateLimit': 1500, // 200 for private API\n            'version': 'v1',\n            // obsolete metainfo interface\n            'hasCORS': false,\n            'hasWithdraw': true,\n            // new metainfo interface\n            'has': {\n                'CORS': false,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27816857-ce7be644-6096-11e7-82d6-3c257263229c.jpg',\n                'api': 'https://api.gemini.com',\n                'www': 'https://gemini.com',\n                'doc': [\n                    'https://docs.gemini.com/rest-api',\n                    'https://docs.sandbox.gemini.com',\n                ],\n                'test': 'https://api.sandbox.gemini.com',\n                'fees': [\n                    'https://gemini.com/fee-schedule/',\n                    'https://gemini.com/transfer-fees/',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'symbols',\n                        'pubticker/{symbol}',\n                        'book/{symbol}',\n                        'trades/{symbol}',\n                        'auction/{symbol}',\n                        'auction/{symbol}/history',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'order/new',\n                        'order/cancel',\n                        'order/cancel/session',\n                        'order/cancel/all',\n                        'order/status',\n                        'orders',\n                        'mytrades',\n                        'tradevolume',\n                        'balances',\n                        'deposit/{currency}/newAddress',\n                        'withdraw/{currency}',\n                        'heartbeat',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetSymbols ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let id = markets[p];\n            let market = id;\n            let uppercase = market.toUpperCase ();\n            let base = uppercase.slice (0, 3);\n            let quote = uppercase.slice (3, 6);\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n                'taker': 0.0025,\n            });\n        }\n        return result;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetBookSymbol (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'amount');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetPubtickerSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        let timestamp = ticker['volume']['timestamp'];\n        let baseVolume = market['base'];\n        let quoteVolume = market['quote'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume'][baseVolume]),\n            'quoteVolume': parseFloat (ticker['volume'][quoteVolume]),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['timestampms'];\n        return {\n            'id': trade['tid'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['type'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTradesSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privatePostBalances ();\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let account = {\n                'free': parseFloat (balance['available']),\n                'used': 0.0,\n                'total': parseFloat (balance['amount']),\n            };\n            account['used'] = account['total'] - account['free'];\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        if (type === 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let nonce = this.nonce ();\n        let order = {\n            'client_order_id': nonce.toString (),\n            'symbol': this.marketId (symbol),\n            'amount': amount.toString (),\n            'price': price.toString (),\n            'side': side,\n            'type': 'exchange limit', // gemini allows limit orders only\n        };\n        let response = await this.privatePostOrderNew (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['order_id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelOrder ({ 'order_id': id });\n    }\n\n    async withdraw (code, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let currency = this.currency (code);\n        let response = await this.privatePostWithdrawCurrency (this.extend ({\n            'currency': currency['id'],\n            'amount': amount,\n            'address': address,\n        }, params));\n        return {\n            'info': response,\n            'id': this.safeString (response, 'txHash'),\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = '/' + this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let request = this.extend ({\n                'request': url,\n                'nonce': nonce,\n            }, query);\n            let payload = this.json (request);\n            payload = this.stringToBase64 (this.encode (payload));\n            let signature = this.hmac (payload, this.encode (this.secret), 'sha384');\n            headers = {\n                'Content-Type': 'text/plain',\n                'X-GEMINI-APIKEY': this.apiKey,\n                'X-GEMINI-PAYLOAD': this.decode (payload),\n                'X-GEMINI-SIGNATURE': signature,\n            };\n        }\n        url = this.urls['api'] + url;\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('result' in response)\n            if (response['result'] === 'error')\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst _1btcxe = require ('./_1btcxe.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class getbtc extends _1btcxe {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'getbtc',\n            'name': 'GetBTC',\n            'countries': [ 'VC', 'RU' ], // Saint Vincent and the Grenadines, Russia, CIS\n            'rateLimit': 1000,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/33801902-03c43462-dd7b-11e7-992e-077e4cd015b9.jpg',\n                'api': 'https://getbtc.org/api',\n                'www': 'https://getbtc.org',\n                'doc': 'https://getbtc.org/api-docs.php',\n            },\n            'markets': {\n                'BTC/EUR': { 'id': 'EUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR', 'precision': { 'amount': 8, 'price': 8 }, 'lot': 0.00000001, 'limits': { 'amount': { 'min': 0.00000001, 'max': undefined }, 'price': { 'min': 0.00000001, 'max': undefined }}},\n                'BTC/RUB': { 'id': 'RUB', 'symbol': 'BTC/RUB', 'base': 'BTC', 'quote': 'RUB', 'precision': { 'amount': 8, 'price': 8 }, 'lot': 0.00000001, 'limits': { 'amount': { 'min': 0.00000001, 'max': undefined }, 'price': { 'min': 0.00000001, 'max': undefined }}},\n                'BTC/USD': { 'id': 'USD', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD', 'precision': { 'amount': 8, 'price': 8 }, 'lot': 0.00000001, 'limits': { 'amount': { 'min': 0.00000001, 'max': undefined }, 'price': { 'min': 0.00000001, 'max': undefined }}},\n            },\n            'fees': {\n                'trading': {\n                    'taker': 0.20 / 100,\n                    'maker': 0.20 / 100,\n                },\n            },\n        });\n    }\n}\n","'use strict';\n\n// ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InsufficientFunds, OrderNotFound } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class hitbtc extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'hitbtc',\n            'name': 'HitBTC',\n            'countries': 'UK',\n            'rateLimit': 1500,\n            'version': '1',\n            'hasCORS': false,\n            'hasFetchTickers': true,\n            'hasFetchOrder': true,\n            'hasFetchOpenOrders': true,\n            'hasFetchClosedOrders': true,\n            'hasWithdraw': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766555-8eaec20e-5edc-11e7-9c5b-6dc69fc42f5e.jpg',\n                'api': 'http://api.hitbtc.com',\n                'www': 'https://hitbtc.com',\n                'doc': 'https://github.com/hitbtc-com/hitbtc-api/blob/master/APIv1.md',\n                'fees': [\n                    'https://hitbtc.com/fees-and-limits',\n                    'https://support.hitbtc.com/hc/en-us/articles/115005148605-Fees-and-limits',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        '{symbol}/orderbook',\n                        '{symbol}/ticker',\n                        '{symbol}/trades',\n                        '{symbol}/trades/recent',\n                        'symbols',\n                        'ticker',\n                        'time',\n                    ],\n                },\n                'trading': {\n                    'get': [\n                        'balance',\n                        'orders/active',\n                        'orders/recent',\n                        'order',\n                        'trades/by/order',\n                        'trades',\n                    ],\n                    'post': [\n                        'new_order',\n                        'cancel_order',\n                        'cancel_orders',\n                    ],\n                },\n                'payment': {\n                    'get': [\n                        'balance',\n                        'address/{currency}',\n                        'transactions',\n                        'transactions/{transaction}',\n                    ],\n                    'post': [\n                        'transfer_to_trading',\n                        'transfer_to_main',\n                        'address/{currency}',\n                        'payout',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'maker': -0.01 / 100,\n                    'taker': 0.1 / 100,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.00085,\n                        'BCC': 0.0018,\n                        'ETH': 0.00215,\n                        'BCH': 0.0018,\n                        'USDT': 100,\n                        'DASH': 0.03,\n                        'BTG': 0.0005,\n                        'LTC': 0.003,\n                        'ZEC': 0.0001,\n                        'XMR': 0.09,\n                        '1ST': 0.84,\n                        'ADX': 5.7,\n                        'AE': 6.7,\n                        'AEON': 0.01006,\n                        'AIR': 565,\n                        'AMP': 9,\n                        'ANT': 6.7,\n                        'ARDR': 1,\n                        'ARN': 18.5,\n                        'ART': 26,\n                        'ATB': 0.0004,\n                        'ATL': 27,\n                        'ATM': 504,\n                        'ATS': 860,\n                        'AVT': 1.9,\n                        'BAS': 113,\n                        'BCN': 0.1,\n                        'BET': 124,\n                        'BKB': 46,\n                        'BMC': 32,\n                        'BMT': 100,\n                        'BNT': 2.57,\n                        'BQX': 4.7,\n                        'BTM': 40,\n                        'BTX': 0.04,\n                        'BUS': 0.004,\n                        'CCT': 115,\n                        'CDT': 100,\n                        'CDX': 30,\n                        'CFI': 61,\n                        'CLD': 0.88,\n                        'CND': 574,\n                        'CNX': 0.04,\n                        'COSS': 65,\n                        'CSNO': 16,\n                        'CTR': 15,\n                        'CTX': 146,\n                        'CVC': 8.46,\n                        'DBIX': 0.0168,\n                        'DCN': 120000,\n                        'DCT': 0.02,\n                        'DDF': 342,\n                        'DENT': 6240,\n                        'DGB': 0.4,\n                        'DGD': 0.01,\n                        'DICE': 0.32,\n                        'DLT': 0.26,\n                        'DNT': 0.21,\n                        'DOGE': 2,\n                        'DOV': 34,\n                        'DRPU': 24,\n                        'DRT': 240,\n                        'DSH': 0.017,\n                        'EBET': 84,\n                        'EBTC': 20,\n                        'EBTCOLD': 6.6,\n                        'ECAT': 14,\n                        'EDG': 2,\n                        'EDO': 2.9,\n                        'ELE': 0.00172,\n                        'ELM': 0.004,\n                        'EMC': 0.03,\n                        'EMGO': 14,\n                        'ENJ': 163,\n                        'EOS': 1.5,\n                        'ERO': 34,\n                        'ETBS': 15,\n                        'ETC': 0.002,\n                        'ETP': 0.004,\n                        'EVX': 5.4,\n                        'EXN': 456,\n                        'FRD': 65,\n                        'FUEL': 123.00105,\n                        'FUN': 202.9598309,\n                        'FYN': 1.849,\n                        'FYP': 66.13,\n                        'GNO': 0.0034,\n                        'GUP': 4,\n                        'GVT': 1.2,\n                        'HAC': 144,\n                        'HDG': 7,\n                        'HGT': 1082,\n                        'HPC': 0.4,\n                        'HVN': 120,\n                        'ICN': 0.55,\n                        'ICO': 34,\n                        'ICOS': 0.35,\n                        'IND': 76,\n                        'INDI': 5913,\n                        'ITS': 15.0012,\n                        'IXT': 11,\n                        'KBR': 143,\n                        'KICK': 112,\n                        'LA': 41,\n                        'LAT': 1.44,\n                        'LIFE': 13000,\n                        'LRC': 27,\n                        'LSK': 0.3,\n                        'LUN': 0.34,\n                        'MAID': 5,\n                        'MANA': 143,\n                        'MCAP': 5.44,\n                        'MIPS': 43,\n                        'MNE': 1.33,\n                        'MSP': 121,\n                        'MTH': 92,\n                        'MYB': 3.9,\n                        'NDC': 165,\n                        'NEBL': 0.04,\n                        'NET': 3.96,\n                        'NTO': 998,\n                        'NXC': 13.39,\n                        'NXT': 3,\n                        'OAX': 15,\n                        'ODN': 0.004,\n                        'OMG': 2,\n                        'OPT': 335,\n                        'ORME': 2.8,\n                        'OTN': 0.57,\n                        'PAY': 3.1,\n                        'PIX': 96,\n                        'PLBT': 0.33,\n                        'PLR': 114,\n                        'PLU': 0.87,\n                        'POE': 784,\n                        'POLL': 3.5,\n                        'PPT': 2,\n                        'PRE': 32,\n                        'PRG': 39,\n                        'PRO': 41,\n                        'PRS': 60,\n                        'PTOY': 0.5,\n                        'QAU': 63,\n                        'QCN': 0.03,\n                        'QTUM': 0.04,\n                        'QVT': 64,\n                        'REP': 0.02,\n                        'RKC': 15,\n                        'RVT': 14,\n                        'SAN': 2.24,\n                        'SBD': 0.03,\n                        'SCL': 2.6,\n                        'SISA': 1640,\n                        'SKIN': 407,\n                        'SMART': 0.4,\n                        'SMS': 0.0375,\n                        'SNC': 36,\n                        'SNGLS': 4,\n                        'SNM': 48,\n                        'SNT': 233,\n                        'STEEM': 0.01,\n                        'STRAT': 0.01,\n                        'STU': 14,\n                        'STX': 11,\n                        'SUB': 17,\n                        'SUR': 3,\n                        'SWT': 0.51,\n                        'TAAS': 0.91,\n                        'TBT': 2.37,\n                        'TFL': 15,\n                        'TIME': 0.03,\n                        'TIX': 7.1,\n                        'TKN': 1,\n                        'TKR': 84,\n                        'TNT': 90,\n                        'TRST': 1.6,\n                        'TRX': 1395,\n                        'UET': 480,\n                        'UGT': 15,\n                        'VEN': 14,\n                        'VERI': 0.037,\n                        'VIB': 50,\n                        'VIBE': 145,\n                        'VOISE': 618,\n                        'WEALTH': 0.0168,\n                        'WINGS': 2.4,\n                        'WTC': 0.75,\n                        'XAUR': 3.23,\n                        'XDN': 0.01,\n                        'XEM': 15,\n                        'XUC': 0.9,\n                        'YOYOW': 140,\n                        'ZAP': 24,\n                        'ZRX': 23,\n                        'ZSC': 191,\n                    },\n                    'deposit': {\n                        'BTC': 0.0006,\n                        'ETH': 0.003,\n                        'BCH': 0,\n                        'USDT': 0,\n                        'BTG': 0,\n                        'LTC': 0,\n                        'ZEC': 0,\n                        'XMR': 0,\n                        '1ST': 0,\n                        'ADX': 0,\n                        'AE': 0,\n                        'AEON': 0,\n                        'AIR': 0,\n                        'AMP': 0,\n                        'ANT': 0,\n                        'ARDR': 0,\n                        'ARN': 0,\n                        'ART': 0,\n                        'ATB': 0,\n                        'ATL': 0,\n                        'ATM': 0,\n                        'ATS': 0,\n                        'AVT': 0,\n                        'BAS': 0,\n                        'BCN': 0,\n                        'BET': 0,\n                        'BKB': 0,\n                        'BMC': 0,\n                        'BMT': 0,\n                        'BNT': 0,\n                        'BQX': 0,\n                        'BTM': 0,\n                        'BTX': 0,\n                        'BUS': 0,\n                        'CCT': 0,\n                        'CDT': 0,\n                        'CDX': 0,\n                        'CFI': 0,\n                        'CLD': 0,\n                        'CND': 0,\n                        'CNX': 0,\n                        'COSS': 0,\n                        'CSNO': 0,\n                        'CTR': 0,\n                        'CTX': 0,\n                        'CVC': 0,\n                        'DBIX': 0,\n                        'DCN': 0,\n                        'DCT': 0,\n                        'DDF': 0,\n                        'DENT': 0,\n                        'DGB': 0,\n                        'DGD': 0,\n                        'DICE': 0,\n                        'DLT': 0,\n                        'DNT': 0,\n                        'DOGE': 0,\n                        'DOV': 0,\n                        'DRPU': 0,\n                        'DRT': 0,\n                        'DSH': 0,\n                        'EBET': 0,\n                        'EBTC': 0,\n                        'EBTCOLD': 0,\n                        'ECAT': 0,\n                        'EDG': 0,\n                        'EDO': 0,\n                        'ELE': 0,\n                        'ELM': 0,\n                        'EMC': 0,\n                        'EMGO': 0,\n                        'ENJ': 0,\n                        'EOS': 0,\n                        'ERO': 0,\n                        'ETBS': 0,\n                        'ETC': 0,\n                        'ETP': 0,\n                        'EVX': 0,\n                        'EXN': 0,\n                        'FRD': 0,\n                        'FUEL': 0,\n                        'FUN': 0,\n                        'FYN': 0,\n                        'FYP': 0,\n                        'GNO': 0,\n                        'GUP': 0,\n                        'GVT': 0,\n                        'HAC': 0,\n                        'HDG': 0,\n                        'HGT': 0,\n                        'HPC': 0,\n                        'HVN': 0,\n                        'ICN': 0,\n                        'ICO': 0,\n                        'ICOS': 0,\n                        'IND': 0,\n                        'INDI': 0,\n                        'ITS': 0,\n                        'IXT': 0,\n                        'KBR': 0,\n                        'KICK': 0,\n                        'LA': 0,\n                        'LAT': 0,\n                        'LIFE': 0,\n                        'LRC': 0,\n                        'LSK': 0,\n                        'LUN': 0,\n                        'MAID': 0,\n                        'MANA': 0,\n                        'MCAP': 0,\n                        'MIPS': 0,\n                        'MNE': 0,\n                        'MSP': 0,\n                        'MTH': 0,\n                        'MYB': 0,\n                        'NDC': 0,\n                        'NEBL': 0,\n                        'NET': 0,\n                        'NTO': 0,\n                        'NXC': 0,\n                        'NXT': 0,\n                        'OAX': 0,\n                        'ODN': 0,\n                        'OMG': 0,\n                        'OPT': 0,\n                        'ORME': 0,\n                        'OTN': 0,\n                        'PAY': 0,\n                        'PIX': 0,\n                        'PLBT': 0,\n                        'PLR': 0,\n                        'PLU': 0,\n                        'POE': 0,\n                        'POLL': 0,\n                        'PPT': 0,\n                        'PRE': 0,\n                        'PRG': 0,\n                        'PRO': 0,\n                        'PRS': 0,\n                        'PTOY': 0,\n                        'QAU': 0,\n                        'QCN': 0,\n                        'QTUM': 0,\n                        'QVT': 0,\n                        'REP': 0,\n                        'RKC': 0,\n                        'RVT': 0,\n                        'SAN': 0,\n                        'SBD': 0,\n                        'SCL': 0,\n                        'SISA': 0,\n                        'SKIN': 0,\n                        'SMART': 0,\n                        'SMS': 0,\n                        'SNC': 0,\n                        'SNGLS': 0,\n                        'SNM': 0,\n                        'SNT': 0,\n                        'STEEM': 0,\n                        'STRAT': 0,\n                        'STU': 0,\n                        'STX': 0,\n                        'SUB': 0,\n                        'SUR': 0,\n                        'SWT': 0,\n                        'TAAS': 0,\n                        'TBT': 0,\n                        'TFL': 0,\n                        'TIME': 0,\n                        'TIX': 0,\n                        'TKN': 0,\n                        'TKR': 0,\n                        'TNT': 0,\n                        'TRST': 0,\n                        'TRX': 0,\n                        'UET': 0,\n                        'UGT': 0,\n                        'VEN': 0,\n                        'VERI': 0,\n                        'VIB': 0,\n                        'VIBE': 0,\n                        'VOISE': 0,\n                        'WEALTH': 0,\n                        'WINGS': 0,\n                        'WTC': 0,\n                        'XAUR': 0,\n                        'XDN': 0,\n                        'XEM': 0,\n                        'XUC': 0,\n                        'YOYOW': 0,\n                        'ZAP': 0,\n                        'ZRX': 0,\n                        'ZSC': 0,\n                    },\n                },\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        if (currency === 'XBT')\n            return 'BTC';\n        if (currency === 'DRK')\n            return 'DASH';\n        if (currency === 'CAT')\n            return 'BitClave';\n        if (currency === 'USD')\n            return 'USDT';\n        return currency;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetSymbols ();\n        let result = [];\n        for (let p = 0; p < markets['symbols'].length; p++) {\n            let market = markets['symbols'][p];\n            let id = market['symbol'];\n            let baseId = market['commodity'];\n            let quoteId = market['currency'];\n            let lot = parseFloat (market['lot']);\n            let step = parseFloat (market['step']);\n            let base = this.commonCurrencyCode (baseId);\n            let quote = this.commonCurrencyCode (quoteId);\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'baseId': baseId,\n                'quoteId': quoteId,\n                'lot': lot,\n                'step': step,\n                'info': market,\n                'precision': {\n                    'amount': this.precisionFromString (market['lot']),\n                    'price': this.precisionFromString (market['step']),\n                },\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': step,\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                },\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let method = this.safeString (params, 'type', 'trading');\n        method += 'GetBalance';\n        let query = this.omit (params, 'type');\n        let response = await this[method] (query);\n        let balances = response['balance'];\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let code = balance['currency_code'];\n            let currency = this.commonCurrencyCode (code);\n            let free = this.safeFloat (balance, 'cash', 0.0);\n            free = this.safeFloat (balance, 'balance', free);\n            let used = this.safeFloat (balance, 'reserved', 0.0);\n            let account = {\n                'free': free,\n                'used': used,\n                'total': this.sum (free, used),\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetSymbolOrderbook (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['timestamp'];\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'bid'),\n            'ask': this.safeFloat (ticker, 'ask'),\n            'vwap': undefined,\n            'open': this.safeFloat (ticker, 'open'),\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'volume'),\n            'quoteVolume': this.safeFloat (ticker, 'volume_quote'),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTicker (params);\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetSymbolTicker (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        if ('message' in ticker)\n            throw new ExchangeError (this.id + ' ' + ticker['message']);\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        return {\n            'info': trade,\n            'id': trade[0],\n            'timestamp': trade[3],\n            'datetime': this.iso8601 (trade[3]),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade[4],\n            'price': parseFloat (trade[1]),\n            'amount': parseFloat (trade[2]),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetSymbolTrades (this.extend ({\n            'symbol': market['id'],\n            // 'from': 0,\n            // 'till': 100,\n            // 'by': 'ts', // or by trade_id\n            // 'sort': 'desc', // or asc\n            // 'start_index': 0,\n            // 'max_results': 1000,\n            // 'format_item': 'object',\n            // 'format_price': 'number',\n            // 'format_amount': 'number',\n            // 'format_tid': 'string',\n            // 'format_timestamp': 'millisecond',\n            // 'format_wrap': false,\n            'side': 'true',\n        }, params));\n        return this.parseTrades (response['trades'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        // check if amount can be evenly divided into lots\n        // they want integer quantity in lot units\n        let quantity = parseFloat (amount) / market['lot'];\n        let wholeLots = Math.round (quantity);\n        let difference = quantity - wholeLots;\n        if (Math.abs (difference) > market['step'])\n            throw new ExchangeError (this.id + ' order amount should be evenly divisible by lot unit size of ' + market['lot'].toString ());\n        let clientOrderId = this.milliseconds ();\n        let order = {\n            'clientOrderId': clientOrderId.toString (),\n            'symbol': market['id'],\n            'side': side,\n            'quantity': wholeLots.toString (), // quantity in integer lot units\n            'type': type,\n        };\n        if (type === 'limit') {\n            order['price'] = this.priceToPrecision (symbol, price);\n        } else {\n            order['timeInForce'] = 'FOK';\n        }\n        let response = await this.tradingPostNewOrder (this.extend (order, params));\n        return this.parseOrder (response['ExecutionReport'], market);\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.tradingPostCancelOrder (this.extend ({\n            'clientOrderId': id,\n        }, params));\n    }\n\n    parseOrderStatus (status) {\n        let statuses = {\n            'new': 'open',\n            'partiallyFilled': 'open',\n            'filled': 'closed',\n            'canceled': 'canceled',\n            'rejected': 'rejected',\n            'expired': 'expired',\n        };\n        return this.safeString (statuses, status);\n    }\n\n    parseOrder (order, market = undefined) {\n        let timestamp = this.safeInteger (order, 'lastTimestamp');\n        if (typeof timestamp === 'undefined')\n            timestamp = this.safeInteger (order, 'timestamp');\n        let symbol = undefined;\n        if (!market)\n            market = this.markets_by_id[order['symbol']];\n        let status = this.safeString (order, 'orderStatus');\n        if (status)\n            status = this.parseOrderStatus (status);\n        let averagePrice = this.safeFloat (order, 'avgPrice', 0.0);\n        let price = this.safeFloat (order, 'orderPrice');\n        if (typeof price === 'undefined')\n            price = this.safeFloat (order, 'price');\n        let amount = this.safeFloat (order, 'orderQuantity');\n        if (typeof amount === 'undefined')\n            amount = this.safeFloat (order, 'quantity');\n        let remaining = this.safeFloat (order, 'quantityLeaves');\n        if (!remaining)\n            remaining = this.safeFloat (order, 'leavesQuantity');\n        let filled = undefined;\n        let cost = undefined;\n        let amountDefined = (typeof amount !== 'undefined');\n        let remainingDefined = (typeof remaining !== 'undefined');\n        if (market) {\n            symbol = market['symbol'];\n            if (amountDefined)\n                amount *= market['lot'];\n            if (remainingDefined)\n                remaining *= market['lot'];\n        }\n        if (amountDefined) {\n            if (remainingDefined) {\n                filled = amount - remaining;\n                cost = averagePrice * filled;\n            }\n        }\n        return {\n            'id': order['clientOrderId'].toString (),\n            'info': order,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': status,\n            'symbol': symbol,\n            'type': order['type'],\n            'side': order['side'],\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'fee': undefined,\n        };\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.tradingGetOrder (this.extend ({\n            'clientOrderId': id,\n        }, params));\n        if (response['orders'][0]) {\n            return this.parseOrder (response['orders'][0]);\n        }\n        throw new OrderNotFound (this.id + ' fetchOrder() error: ' + this.response);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let statuses = [ 'new', 'partiallyFiiled' ];\n        let market = undefined;\n        let request = {\n            'sort': 'desc',\n            'statuses': statuses.join (','),\n        };\n        if (symbol) {\n            market = this.market (symbol);\n            request['symbols'] = market['id'];\n        }\n        let response = await this.tradingGetOrdersActive (this.extend (request, params));\n        return this.parseOrders (response['orders'], market, since, limit);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        let statuses = [ 'filled', 'canceled', 'rejected', 'expired' ];\n        let request = {\n            'sort': 'desc',\n            'statuses': statuses.join (','),\n            'max_results': 1000,\n        };\n        if (symbol) {\n            market = this.market (symbol);\n            request['symbols'] = market['id'];\n        }\n        let response = await this.tradingGetOrdersRecent (this.extend (request, params));\n        return this.parseOrders (response['orders'], market, since, limit);\n    }\n\n    async withdraw (code, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let currency = this.currency (code);\n        let request = {\n            'currency_code': currency['id'],\n            'amount': amount,\n            'address': address,\n        };\n        if (tag)\n            request['paymentId'] = tag;\n        let response = await this.paymentPostPayout (this.extend (request, params));\n        return {\n            'info': response,\n            'id': response['transaction'],\n        };\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = '/' + 'api' + '/' + this.version + '/' + api + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let payload = { 'nonce': nonce, 'apikey': this.apiKey };\n            query = this.extend (payload, query);\n            if (method === 'GET')\n                url += '?' + this.urlencode (query);\n            else\n                url += '?' + this.urlencode (payload);\n            let auth = url;\n            if (method === 'POST') {\n                if (Object.keys (query).length) {\n                    body = this.urlencode (query);\n                    auth += body;\n                }\n            }\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'X-Signature': this.hmac (this.encode (auth), this.encode (this.secret), 'sha512').toLowerCase (),\n            };\n        }\n        url = this.urls['api'] + url;\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('code' in response) {\n            if ('ExecutionReport' in response) {\n                if (response['ExecutionReport']['orderRejectReason'] === 'orderExceedsLimit')\n                    throw new InsufficientFunds (this.id + ' ' + this.json (response));\n            }\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        return response;\n    }\n};\n","'use strict';\n\n// ---------------------------------------------------------------------------\n\nconst hitbtc = require ('./hitbtc');\nconst { ExchangeError, OrderNotFound, InsufficientFunds, InvalidOrder } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class hitbtc2 extends hitbtc {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'hitbtc2',\n            'name': 'HitBTC v2',\n            'countries': 'UK',\n            'rateLimit': 1500,\n            'version': '2',\n            'hasCORS': true,\n            // older metainfo interface\n            'hasFetchOHLCV': true,\n            'hasFetchTickers': true,\n            'hasFetchOrder': true,\n            'hasFetchOrders': false,\n            'hasFetchOpenOrders': true,\n            'hasFetchClosedOrders': true,\n            'hasFetchMyTrades': true,\n            'hasWithdraw': true,\n            'hasFetchCurrencies': true,\n            // new metainfo interface\n            'has': {\n                'fetchCurrencies': true,\n                'fetchOHLCV': true,\n                'fetchTickers': true,\n                'fetchOrder': true,\n                'fetchOrders': false,\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': true,\n                'fetchMyTrades': true,\n                'withdraw': true,\n            },\n            'timeframes': {\n                '1m': 'M1',\n                '3m': 'M3',\n                '5m': 'M5',\n                '15m': 'M15',\n                '30m': 'M30', // default\n                '1h': 'H1',\n                '4h': 'H4',\n                '1d': 'D1',\n                '1w': 'D7',\n                '1M': '1M',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766555-8eaec20e-5edc-11e7-9c5b-6dc69fc42f5e.jpg',\n                'api': 'https://api.hitbtc.com',\n                'www': 'https://hitbtc.com',\n                'doc': 'https://api.hitbtc.com',\n                'fees': [\n                    'https://hitbtc.com/fees-and-limits',\n                    'https://support.hitbtc.com/hc/en-us/articles/115005148605-Fees-and-limits',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'symbol', // Available Currency Symbols\n                        'symbol/{symbol}', // Get symbol info\n                        'currency', // Available Currencies\n                        'currency/{currency}', // Get currency info\n                        'ticker', // Ticker list for all symbols\n                        'ticker/{symbol}', // Ticker for symbol\n                        'trades/{symbol}', // Trades\n                        'orderbook/{symbol}', // Orderbook\n                        'candles/{symbol}', // Candles\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'order', // List your current open orders\n                        'order/{clientOrderId}', // Get a single order by clientOrderId\n                        'trading/balance', // Get trading balance\n                        'trading/fee/{symbol}', // Get trading fee rate\n                        'history/trades', // Get historical trades\n                        'history/order', // Get historical orders\n                        'history/order/{id}/trades', // Get historical trades by specified order\n                        'account/balance', // Get main acccount balance\n                        'account/transactions', // Get account transactions\n                        'account/transactions/{id}', // Get account transaction by id\n                        'account/crypto/address/{currency}', // Get deposit crypro address\n                    ],\n                    'post': [\n                        'order', // Create new order\n                        'account/crypto/withdraw', // Withdraw crypro\n                        'account/crypto/address/{currency}', // Create new deposit crypro address\n                        'account/transfer', // Transfer amount to trading\n                    ],\n                    'put': [\n                        'order/{clientOrderId}', // Create new order\n                        'account/crypto/withdraw/{id}', // Commit withdraw crypro\n                    ],\n                    'delete': [\n                        'order', // Cancel all open orders\n                        'order/{clientOrderId}', // Cancel order\n                        'account/crypto/withdraw/{id}', // Rollback withdraw crypro\n                    ],\n                    'patch': [\n                        'order/{clientOrderId}', // Cancel Replace order\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'maker': -0.01 / 100,\n                    'taker': 0.1 / 100,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.00085,\n                        'BCC': 0.0018,\n                        'ETH': 0.00215,\n                        'BCH': 0.0018,\n                        'USDT': 100,\n                        'DASH': 0.03,\n                        'BTG': 0.0005,\n                        'LTC': 0.003,\n                        'ZEC': 0.0001,\n                        'XMR': 0.09,\n                        '1ST': 0.84,\n                        'ADX': 5.7,\n                        'AE': 6.7,\n                        'AEON': 0.01006,\n                        'AIR': 565,\n                        'AMP': 9,\n                        'ANT': 6.7,\n                        'ARDR': 1,\n                        'ARN': 18.5,\n                        'ART': 26,\n                        'ATB': 0.0004,\n                        'ATL': 27,\n                        'ATM': 504,\n                        'ATS': 860,\n                        'AVT': 1.9,\n                        'BAS': 113,\n                        'BCN': 0.1,\n                        'BET': 124,\n                        'BKB': 46,\n                        'BMC': 32,\n                        'BMT': 100,\n                        'BNT': 2.57,\n                        'BQX': 4.7,\n                        'BTM': 40,\n                        'BTX': 0.04,\n                        'BUS': 0.004,\n                        'CCT': 115,\n                        'CDT': 100,\n                        'CDX': 30,\n                        'CFI': 61,\n                        'CLD': 0.88,\n                        'CND': 574,\n                        'CNX': 0.04,\n                        'COSS': 65,\n                        'CSNO': 16,\n                        'CTR': 15,\n                        'CTX': 146,\n                        'CVC': 8.46,\n                        'DBIX': 0.0168,\n                        'DCN': 120000,\n                        'DCT': 0.02,\n                        'DDF': 342,\n                        'DENT': 6240,\n                        'DGB': 0.4,\n                        'DGD': 0.01,\n                        'DICE': 0.32,\n                        'DLT': 0.26,\n                        'DNT': 0.21,\n                        'DOGE': 2,\n                        'DOV': 34,\n                        'DRPU': 24,\n                        'DRT': 240,\n                        'DSH': 0.017,\n                        'EBET': 84,\n                        'EBTC': 20,\n                        'EBTCOLD': 6.6,\n                        'ECAT': 14,\n                        'EDG': 2,\n                        'EDO': 2.9,\n                        'ELE': 0.00172,\n                        'ELM': 0.004,\n                        'EMC': 0.03,\n                        'EMGO': 14,\n                        'ENJ': 163,\n                        'EOS': 1.5,\n                        'ERO': 34,\n                        'ETBS': 15,\n                        'ETC': 0.002,\n                        'ETP': 0.004,\n                        'EVX': 5.4,\n                        'EXN': 456,\n                        'FRD': 65,\n                        'FUEL': 123.00105,\n                        'FUN': 202.9598309,\n                        'FYN': 1.849,\n                        'FYP': 66.13,\n                        'GNO': 0.0034,\n                        'GUP': 4,\n                        'GVT': 1.2,\n                        'HAC': 144,\n                        'HDG': 7,\n                        'HGT': 1082,\n                        'HPC': 0.4,\n                        'HVN': 120,\n                        'ICN': 0.55,\n                        'ICO': 34,\n                        'ICOS': 0.35,\n                        'IND': 76,\n                        'INDI': 5913,\n                        'ITS': 15.0012,\n                        'IXT': 11,\n                        'KBR': 143,\n                        'KICK': 112,\n                        'LA': 41,\n                        'LAT': 1.44,\n                        'LIFE': 13000,\n                        'LRC': 27,\n                        'LSK': 0.3,\n                        'LUN': 0.34,\n                        'MAID': 5,\n                        'MANA': 143,\n                        'MCAP': 5.44,\n                        'MIPS': 43,\n                        'MNE': 1.33,\n                        'MSP': 121,\n                        'MTH': 92,\n                        'MYB': 3.9,\n                        'NDC': 165,\n                        'NEBL': 0.04,\n                        'NET': 3.96,\n                        'NTO': 998,\n                        'NXC': 13.39,\n                        'NXT': 3,\n                        'OAX': 15,\n                        'ODN': 0.004,\n                        'OMG': 2,\n                        'OPT': 335,\n                        'ORME': 2.8,\n                        'OTN': 0.57,\n                        'PAY': 3.1,\n                        'PIX': 96,\n                        'PLBT': 0.33,\n                        'PLR': 114,\n                        'PLU': 0.87,\n                        'POE': 784,\n                        'POLL': 3.5,\n                        'PPT': 2,\n                        'PRE': 32,\n                        'PRG': 39,\n                        'PRO': 41,\n                        'PRS': 60,\n                        'PTOY': 0.5,\n                        'QAU': 63,\n                        'QCN': 0.03,\n                        'QTUM': 0.04,\n                        'QVT': 64,\n                        'REP': 0.02,\n                        'RKC': 15,\n                        'RVT': 14,\n                        'SAN': 2.24,\n                        'SBD': 0.03,\n                        'SCL': 2.6,\n                        'SISA': 1640,\n                        'SKIN': 407,\n                        'SMART': 0.4,\n                        'SMS': 0.0375,\n                        'SNC': 36,\n                        'SNGLS': 4,\n                        'SNM': 48,\n                        'SNT': 233,\n                        'STEEM': 0.01,\n                        'STRAT': 0.01,\n                        'STU': 14,\n                        'STX': 11,\n                        'SUB': 17,\n                        'SUR': 3,\n                        'SWT': 0.51,\n                        'TAAS': 0.91,\n                        'TBT': 2.37,\n                        'TFL': 15,\n                        'TIME': 0.03,\n                        'TIX': 7.1,\n                        'TKN': 1,\n                        'TKR': 84,\n                        'TNT': 90,\n                        'TRST': 1.6,\n                        'TRX': 1395,\n                        'UET': 480,\n                        'UGT': 15,\n                        'VEN': 14,\n                        'VERI': 0.037,\n                        'VIB': 50,\n                        'VIBE': 145,\n                        'VOISE': 618,\n                        'WEALTH': 0.0168,\n                        'WINGS': 2.4,\n                        'WTC': 0.75,\n                        'XAUR': 3.23,\n                        'XDN': 0.01,\n                        'XEM': 15,\n                        'XUC': 0.9,\n                        'YOYOW': 140,\n                        'ZAP': 24,\n                        'ZRX': 23,\n                        'ZSC': 191,\n                    },\n                    'deposit': {\n                        'BTC': 0.0006,\n                        'ETH': 0.003,\n                        'BCH': 0,\n                        'USDT': 0,\n                        'BTG': 0,\n                        'LTC': 0,\n                        'ZEC': 0,\n                        'XMR': 0,\n                        '1ST': 0,\n                        'ADX': 0,\n                        'AE': 0,\n                        'AEON': 0,\n                        'AIR': 0,\n                        'AMP': 0,\n                        'ANT': 0,\n                        'ARDR': 0,\n                        'ARN': 0,\n                        'ART': 0,\n                        'ATB': 0,\n                        'ATL': 0,\n                        'ATM': 0,\n                        'ATS': 0,\n                        'AVT': 0,\n                        'BAS': 0,\n                        'BCN': 0,\n                        'BET': 0,\n                        'BKB': 0,\n                        'BMC': 0,\n                        'BMT': 0,\n                        'BNT': 0,\n                        'BQX': 0,\n                        'BTM': 0,\n                        'BTX': 0,\n                        'BUS': 0,\n                        'CCT': 0,\n                        'CDT': 0,\n                        'CDX': 0,\n                        'CFI': 0,\n                        'CLD': 0,\n                        'CND': 0,\n                        'CNX': 0,\n                        'COSS': 0,\n                        'CSNO': 0,\n                        'CTR': 0,\n                        'CTX': 0,\n                        'CVC': 0,\n                        'DBIX': 0,\n                        'DCN': 0,\n                        'DCT': 0,\n                        'DDF': 0,\n                        'DENT': 0,\n                        'DGB': 0,\n                        'DGD': 0,\n                        'DICE': 0,\n                        'DLT': 0,\n                        'DNT': 0,\n                        'DOGE': 0,\n                        'DOV': 0,\n                        'DRPU': 0,\n                        'DRT': 0,\n                        'DSH': 0,\n                        'EBET': 0,\n                        'EBTC': 0,\n                        'EBTCOLD': 0,\n                        'ECAT': 0,\n                        'EDG': 0,\n                        'EDO': 0,\n                        'ELE': 0,\n                        'ELM': 0,\n                        'EMC': 0,\n                        'EMGO': 0,\n                        'ENJ': 0,\n                        'EOS': 0,\n                        'ERO': 0,\n                        'ETBS': 0,\n                        'ETC': 0,\n                        'ETP': 0,\n                        'EVX': 0,\n                        'EXN': 0,\n                        'FRD': 0,\n                        'FUEL': 0,\n                        'FUN': 0,\n                        'FYN': 0,\n                        'FYP': 0,\n                        'GNO': 0,\n                        'GUP': 0,\n                        'GVT': 0,\n                        'HAC': 0,\n                        'HDG': 0,\n                        'HGT': 0,\n                        'HPC': 0,\n                        'HVN': 0,\n                        'ICN': 0,\n                        'ICO': 0,\n                        'ICOS': 0,\n                        'IND': 0,\n                        'INDI': 0,\n                        'ITS': 0,\n                        'IXT': 0,\n                        'KBR': 0,\n                        'KICK': 0,\n                        'LA': 0,\n                        'LAT': 0,\n                        'LIFE': 0,\n                        'LRC': 0,\n                        'LSK': 0,\n                        'LUN': 0,\n                        'MAID': 0,\n                        'MANA': 0,\n                        'MCAP': 0,\n                        'MIPS': 0,\n                        'MNE': 0,\n                        'MSP': 0,\n                        'MTH': 0,\n                        'MYB': 0,\n                        'NDC': 0,\n                        'NEBL': 0,\n                        'NET': 0,\n                        'NTO': 0,\n                        'NXC': 0,\n                        'NXT': 0,\n                        'OAX': 0,\n                        'ODN': 0,\n                        'OMG': 0,\n                        'OPT': 0,\n                        'ORME': 0,\n                        'OTN': 0,\n                        'PAY': 0,\n                        'PIX': 0,\n                        'PLBT': 0,\n                        'PLR': 0,\n                        'PLU': 0,\n                        'POE': 0,\n                        'POLL': 0,\n                        'PPT': 0,\n                        'PRE': 0,\n                        'PRG': 0,\n                        'PRO': 0,\n                        'PRS': 0,\n                        'PTOY': 0,\n                        'QAU': 0,\n                        'QCN': 0,\n                        'QTUM': 0,\n                        'QVT': 0,\n                        'REP': 0,\n                        'RKC': 0,\n                        'RVT': 0,\n                        'SAN': 0,\n                        'SBD': 0,\n                        'SCL': 0,\n                        'SISA': 0,\n                        'SKIN': 0,\n                        'SMART': 0,\n                        'SMS': 0,\n                        'SNC': 0,\n                        'SNGLS': 0,\n                        'SNM': 0,\n                        'SNT': 0,\n                        'STEEM': 0,\n                        'STRAT': 0,\n                        'STU': 0,\n                        'STX': 0,\n                        'SUB': 0,\n                        'SUR': 0,\n                        'SWT': 0,\n                        'TAAS': 0,\n                        'TBT': 0,\n                        'TFL': 0,\n                        'TIME': 0,\n                        'TIX': 0,\n                        'TKN': 0,\n                        'TKR': 0,\n                        'TNT': 0,\n                        'TRST': 0,\n                        'TRX': 0,\n                        'UET': 0,\n                        'UGT': 0,\n                        'VEN': 0,\n                        'VERI': 0,\n                        'VIB': 0,\n                        'VIBE': 0,\n                        'VOISE': 0,\n                        'WEALTH': 0,\n                        'WINGS': 0,\n                        'WTC': 0,\n                        'XAUR': 0,\n                        'XDN': 0,\n                        'XEM': 0,\n                        'XUC': 0,\n                        'YOYOW': 0,\n                        'ZAP': 0,\n                        'ZRX': 0,\n                        'ZSC': 0,\n                    },\n                },\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        let currencies = {\n            'XBT': 'BTC',\n            'DRK': 'DASH',\n            'CAT': 'BitClave',\n            'USD': 'USDT',\n        };\n        if (currency in currencies)\n            return currencies[currency];\n        return currency;\n    }\n\n    feeToPrecision (symbol, fee) {\n        return this.truncate (fee, 8);\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetSymbol ();\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let id = market['id'];\n            let baseId = market['baseCurrency'];\n            let quoteId = market['quoteCurrency'];\n            let base = this.commonCurrencyCode (baseId);\n            let quote = this.commonCurrencyCode (quoteId);\n            let symbol = base + '/' + quote;\n            let lot = parseFloat (market['quantityIncrement']);\n            let step = parseFloat (market['tickSize']);\n            let precision = {\n                'price': this.precisionFromString (market['tickSize']),\n                'amount': this.precisionFromString (market['quantityIncrement']),\n            };\n            let taker = parseFloat (market['takeLiquidityRate']);\n            let maker = parseFloat (market['provideLiquidityRate']);\n            result.push (this.extend (this.fees['trading'], {\n                'info': market,\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'baseId': baseId,\n                'quoteId': quoteId,\n                'active': true,\n                'lot': lot,\n                'step': step,\n                'taker': taker,\n                'maker': maker,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': step,\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': lot * step,\n                        'max': undefined,\n                    },\n                },\n            }));\n        }\n        return result;\n    }\n\n    async fetchCurrencies (params = {}) {\n        let currencies = await this.publicGetCurrency (params);\n        let result = {};\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let id = currency['id'];\n            // todo: will need to rethink the fees\n            // to add support for multiple withdrawal/deposit methods and\n            // differentiated fees for each particular method\n            let precision = 8; // default precision, todo: fix \"magic constants\"\n            let code = this.commonCurrencyCode (id);\n            let payin = currency['payinEnabled'];\n            let payout = currency['payoutEnabled'];\n            let transfer = currency['transferEnabled'];\n            let active = payin && payout && transfer;\n            let status = 'ok';\n            if ('disabled' in currency)\n                if (currency['disabled'])\n                    status = 'disabled';\n            let type = (currency['crypto']) ? 'crypto' : 'fiat';\n            result[code] = {\n                'id': id,\n                'code': code,\n                'type': type,\n                'payin': payin,\n                'payout': payout,\n                'transfer': transfer,\n                'info': currency,\n                'name': currency['fullName'],\n                'active': active,\n                'status': status,\n                'fee': undefined, // todo: redesign\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': undefined,\n                        'max': Math.pow (10, precision),\n                    },\n                },\n            };\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let type = this.safeString (params, 'type', 'trading');\n        let method = 'privateGet' + this.capitalize (type) + 'Balance';\n        let balances = await this[method] ();\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let code = balance['currency'];\n            let currency = this.commonCurrencyCode (code);\n            let account = {\n                'free': parseFloat (balance['available']),\n                'used': parseFloat (balance['reserved']),\n                'total': 0.0,\n            };\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1d', since = undefined, limit = undefined) {\n        let timestamp = this.parse8601 (ohlcv['timestamp']);\n        return [\n            timestamp,\n            parseFloat (ohlcv['open']),\n            parseFloat (ohlcv['max']),\n            parseFloat (ohlcv['min']),\n            parseFloat (ohlcv['close']),\n            parseFloat (ohlcv['volumeQuote']),\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n            'period': this.timeframes[timeframe],\n        };\n        if (limit)\n            request['limit'] = limit;\n        let response = await this.publicGetCandlesSymbol (this.extend (request, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderbookSymbol (this.extend ({\n            'symbol': this.marketId (symbol),\n            // 'limit': 100, // default = 100, 0 = unlimited\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bid', 'ask', 'price', 'size');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.parse8601 (ticker['timestamp']);\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'bid'),\n            'ask': this.safeFloat (ticker, 'ask'),\n            'vwap': undefined,\n            'open': this.safeFloat (ticker, 'open'),\n            'close': this.safeFloat (ticker, 'close'),\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'volume'),\n            'quoteVolume': this.safeFloat (ticker, 'volumeQuote'),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTicker (params);\n        let result = {};\n        for (let i = 0; i < tickers.length; i++) {\n            let ticker = tickers[i];\n            let id = ticker['symbol'];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetTickerSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        if ('message' in ticker)\n            throw new ExchangeError (this.id + ' ' + ticker['message']);\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = this.parse8601 (trade['timestamp']);\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else {\n            let id = trade['symbol'];\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            } else {\n                symbol = id;\n            }\n        }\n        let fee = undefined;\n        if ('fee' in trade) {\n            let currency = market ? market['quote'] : undefined;\n            fee = {\n                'cost': parseFloat (trade['fee']),\n                'currency': currency,\n            };\n        }\n        let orderId = undefined;\n        if ('clientOrderId' in trade)\n            orderId = trade['clientOrderId'];\n        let price = parseFloat (trade['price']);\n        let amount = parseFloat (trade['quantity']);\n        let cost = price * amount;\n        return {\n            'info': trade,\n            'id': trade['id'].toString (),\n            'order': orderId,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': undefined,\n            'side': trade['side'],\n            'price': price,\n            'amount': amount,\n            'cost': cost,\n            'fee': fee,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTradesSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        // their max accepted length is 32 characters\n        let uuid = this.uuid ();\n        let parts = uuid.split ('-');\n        let clientOrderId = parts.join ('');\n        clientOrderId = clientOrderId.slice (0, 32);\n        amount = parseFloat (amount);\n        let request = {\n            'clientOrderId': clientOrderId,\n            'symbol': market['id'],\n            'side': side,\n            'quantity': this.amountToPrecision (symbol, amount),\n            'type': type,\n        };\n        if (type === 'limit') {\n            request['price'] = this.priceToPrecision (symbol, price);\n        } else {\n            request['timeInForce'] = 'FOK';\n        }\n        let response = await this.privatePostOrder (this.extend (request, params));\n        let order = this.parseOrder (response);\n        let id = order['id'];\n        this.orders[id] = order;\n        return order;\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privateDeleteOrderClientOrderId (this.extend ({\n            'clientOrderId': id,\n        }, params));\n    }\n\n    parseOrder (order, market = undefined) {\n        let created = undefined;\n        if ('createdAt' in order)\n            created = this.parse8601 (order['createdAt']);\n        let updated = undefined;\n        if ('updatedAt' in order)\n            updated = this.parse8601 (order['updatedAt']);\n        if (!market)\n            market = this.markets_by_id[order['symbol']];\n        let symbol = market['symbol'];\n        let amount = this.safeFloat (order, 'quantity');\n        let filled = this.safeFloat (order, 'cumQuantity');\n        let status = order['status'];\n        if (status === 'new') {\n            status = 'open';\n        } else if (status === 'suspended') {\n            status = 'open';\n        } else if (status === 'partiallyFilled') {\n            status = 'open';\n        } else if (status === 'filled') {\n            status = 'closed';\n        }\n        let id = order['clientOrderId'].toString ();\n        let price = this.safeFloat (order, 'price');\n        if (typeof price === 'undefined') {\n            if (id in this.orders)\n                price = this.orders[id]['price'];\n        }\n        let remaining = undefined;\n        let cost = undefined;\n        if (typeof amount !== 'undefined') {\n            if (typeof filled !== 'undefined') {\n                remaining = amount - filled;\n                if (typeof price !== 'undefined') {\n                    cost = filled * price;\n                }\n            }\n        }\n        return {\n            'id': id,\n            'timestamp': created,\n            'datetime': this.iso8601 (created),\n            'created': created,\n            'updated': updated,\n            'status': status,\n            'symbol': symbol,\n            'type': order['type'],\n            'side': order['side'],\n            'price': price,\n            'amount': amount,\n            'cost': cost,\n            'filled': filled,\n            'remaining': remaining,\n            'fee': undefined,\n            'info': order,\n        };\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetHistoryOrder (this.extend ({\n            'clientOrderId': id,\n        }, params));\n        let numOrders = response.length;\n        if (numOrders > 0)\n            return this.parseOrder (response[0]);\n        throw new OrderNotFound (this.id + ' order ' + id + ' not found');\n    }\n\n    async fetchOpenOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetOrderClientOrderId (this.extend ({\n            'clientOrderId': id,\n        }, params));\n        return this.parseOrder (response);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        let request = {};\n        if (symbol) {\n            market = this.market (symbol);\n            request['symbol'] = market['id'];\n        }\n        let response = await this.privateGetOrder (this.extend (request, params));\n        return this.parseOrders (response, market, since, limit);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        let request = {};\n        if (symbol) {\n            market = this.market (symbol);\n            request['symbol'] = market['id'];\n        }\n        if (limit)\n            request['limit'] = limit;\n        if (since) {\n            request['from'] = this.iso8601 (since);\n        }\n        let response = await this.privateGetHistoryOrder (this.extend (request, params));\n        return this.parseOrders (response, market, since, limit);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            // 'symbol': 'BTC/USD', // optional\n            // 'sort': 'DESC', // or 'ASC'\n            // 'by': 'timestamp', // or 'id'\tString\ttimestamp by default, or id\n            // 'from':\t'Datetime or Number', // ISO 8601\n            // 'till':\t'Datetime or Number',\n            // 'limit': 100,\n            // 'offset': 0,\n        };\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['symbol'] = market['id'];\n        }\n        if (since)\n            request['from'] = this.iso8601 (since);\n        if (limit)\n            request['limit'] = limit;\n        let response = await this.privateGetHistoryTrades (this.extend (request, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchOrderTrades (id, symbol = undefined, params = {}) {\n        // The id needed here is the exchange's id, and not the clientOrderID, which is\n        // the id that is stored in the unified api order id. In order the get the exchange's id,\n        // you need to grab it from order['info']['id']\n        await this.loadMarkets ();\n        let trades = await this.privateGetHistoryOrderIdTrades (this.extend ({\n            'id': id,\n        }, params));\n        return this.parseTrades (trades);\n    }\n\n    async createDepositAddress (code, params = {}) {\n        await this.loadMarkets ();\n        let currency = this.currency (code);\n        let response = await this.privatePostAccountCryptoAddressCurrency ({\n            'currency': currency['id'],\n        });\n        let address = response['address'];\n        return {\n            'currency': currency,\n            'address': address,\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    async fetchDepositAddress (code, params = {}) {\n        await this.loadMarkets ();\n        let currency = this.currency (code);\n        let response = await this.privateGetAccountCryptoAddressCurrency ({\n            'currency': currency['id'],\n        });\n        let address = response['address'];\n        return {\n            'currency': currency,\n            'address': address,\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    async withdraw (code, amount, address, tag = undefined, params = {}) {\n        let currency = this.currency (code);\n        let request = {\n            'currency': currency['id'],\n            'amount': parseFloat (amount),\n            'address': address,\n        };\n        if (tag)\n            request['paymentId'] = tag;\n        let response = await this.privatePostAccountCryptoWithdraw (this.extend (request, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = '/api' + '/' + this.version + '/';\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            url += api + '/' + this.implodeParams (path, params);\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            url += this.implodeParams (path, params);\n            if (method === 'GET') {\n                if (Object.keys (query).length)\n                    url += '?' + this.urlencode (query);\n            } else {\n                if (Object.keys (query).length)\n                    body = this.json (query);\n            }\n            let payload = this.encode (this.apiKey + ':' + this.secret);\n            let auth = this.stringToBase64 (payload);\n            headers = {\n                'Authorization': 'Basic ' + this.decode (auth),\n                'Content-Type': 'application/json',\n            };\n        }\n        url = this.urls['api'] + url;\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code === 400) {\n            if (body[0] === '{') {\n                let response = JSON.parse (body);\n                if ('error' in response) {\n                    if ('message' in response['error']) {\n                        let message = response['error']['message'];\n                        if (message === 'Order not found') {\n                            throw new OrderNotFound (this.id + ' order not found in active orders');\n                        } else if (message === 'Insufficient funds') {\n                            throw new InsufficientFunds (this.id + ' ' + body);\n                        } else if (message === 'Duplicate clientOrderId') {\n                            throw new InvalidOrder (this.id + ' ' + body);\n                        }\n                    }\n                }\n            }\n            throw new ExchangeError (this.id + ' ' + body);\n        }\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class huobi extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'huobi',\n            'name': 'Huobi',\n            'countries': 'CN',\n            'rateLimit': 2000,\n            'version': 'v3',\n            'hasCORS': false,\n            'hasFetchOHLCV': true,\n            'timeframes': {\n                '1m': '001',\n                '5m': '005',\n                '15m': '015',\n                '30m': '030',\n                '1h': '060',\n                '1d': '100',\n                '1w': '200',\n                '1M': '300',\n                '1y': '400',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766569-15aa7b9a-5edd-11e7-9e7f-44791f4ee49c.jpg',\n                'api': 'http://api.huobi.com',\n                'www': 'https://www.huobi.com',\n                'doc': 'https://github.com/huobiapi/API_Docs_en/wiki',\n            },\n            'api': {\n                'staticmarket': {\n                    'get': [\n                        '{id}_kline_{period}',\n                        'ticker_{id}',\n                        'depth_{id}',\n                        'depth_{id}_{length}',\n                        'detail_{id}',\n                    ],\n                },\n                'usdmarket': {\n                    'get': [\n                        '{id}_kline_{period}',\n                        'ticker_{id}',\n                        'depth_{id}',\n                        'depth_{id}_{length}',\n                        'detail_{id}',\n                    ],\n                },\n                'trade': {\n                    'post': [\n                        'get_account_info',\n                        'get_orders',\n                        'order_info',\n                        'buy',\n                        'sell',\n                        'buy_market',\n                        'sell_market',\n                        'cancel_order',\n                        'get_new_deal_orders',\n                        'get_order_id_by_trade_id',\n                        'withdraw_coin',\n                        'cancel_withdraw_coin',\n                        'get_withdraw_coin_result',\n                        'transfer',\n                        'loan',\n                        'repayment',\n                        'get_loan_available',\n                        'get_loans',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/CNY': { 'id': 'btc', 'symbol': 'BTC/CNY', 'base': 'BTC', 'quote': 'CNY', 'type': 'staticmarket', 'coinType': 1 },\n                'LTC/CNY': { 'id': 'ltc', 'symbol': 'LTC/CNY', 'base': 'LTC', 'quote': 'CNY', 'type': 'staticmarket', 'coinType': 2 },\n                // 'BTC/USD': { 'id': 'btc', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD', 'type': 'usdmarket',    'coinType': 1 },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let balances = await this.tradePostGetAccountInfo ();\n        let result = { 'info': balances };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let account = this.account ();\n            let available = 'available_' + lowercase + '_display';\n            let frozen = 'frozen_' + lowercase + '_display';\n            let loan = 'loan_' + lowercase + '_display';\n            if (available in balances)\n                account['free'] = parseFloat (balances[available]);\n            if (frozen in balances)\n                account['used'] = parseFloat (balances[frozen]);\n            if (loan in balances)\n                account['used'] = this.sum (account['used'], parseFloat (balances[loan]));\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let market = this.market (symbol);\n        let method = market['type'] + 'GetDepthId';\n        let orderbook = await this[method] (this.extend ({ 'id': market['id'] }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let market = this.market (symbol);\n        let method = market['type'] + 'GetTickerId';\n        let response = await this[method] (this.extend ({\n            'id': market['id'],\n        }, params));\n        let ticker = response['ticker'];\n        let timestamp = parseInt (response['time']) * 1000;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'buy'),\n            'ask': this.safeFloat (ticker, 'sell'),\n            'vwap': undefined,\n            'open': this.safeFloat (ticker, 'open'),\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': this.safeFloat (ticker, 'vol'),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['ts'];\n        return {\n            'info': trade,\n            'id': trade['id'].toString (),\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['direction'],\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let method = market['type'] + 'GetDetailId';\n        let response = await this[method] (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTrades (response['trades'], market, since, limit);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        // not implemented yet\n        return [\n            ohlcv[0],\n            ohlcv[1],\n            ohlcv[2],\n            ohlcv[3],\n            ohlcv[4],\n            ohlcv[6],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let method = market['type'] + 'GetIdKlinePeriod';\n        let ohlcvs = await this[method] (this.extend ({\n            'id': market['id'],\n            'period': this.timeframes[timeframe],\n        }, params));\n        return ohlcvs;\n        // return this.parseOHLCVs (ohlcvs, market, timeframe, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let market = this.market (symbol);\n        let method = 'tradePost' + this.capitalize (side);\n        let order = {\n            'coin_type': market['coinType'],\n            'amount': amount,\n            'market': market['quote'].toLowerCase (),\n        };\n        if (type == 'limit')\n            order['price'] = price;\n        else\n            method += this.capitalize (type);\n        let response = this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.tradePostCancelOrder ({ 'id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'];\n        if (api == 'trade') {\n            this.checkRequiredCredentials ();\n            url += '/api' + this.version;\n            let query = this.keysort (this.extend ({\n                'method': path,\n                'access_key': this.apiKey,\n                'created': this.nonce (),\n            }, params));\n            let queryString = this.urlencode (this.omit (query, 'market'));\n            // secret key must be appended to the query before signing\n            queryString += '&secret_key=' + this.secret;\n            query['sign'] = this.hash (this.encode (queryString));\n            body = this.urlencode (query);\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        } else {\n            url += '/' + api + '/' + this.implodeParams (path, params) + '_json.js';\n            let query = this.omit (params, this.extractParams (path));\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'trade', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('status' in response)\n            if (response['status'] == 'error')\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        if ('code' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst huobipro = require ('./huobipro.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class huobicny extends huobipro {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'huobicny',\n            'name': 'Huobi CNY',\n            'hostname': 'be.huobi.com',\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766569-15aa7b9a-5edd-11e7-9e7f-44791f4ee49c.jpg',\n                'api': 'https://be.huobi.com',\n                'www': 'https://www.huobi.com',\n                'doc': 'https://github.com/huobiapi/API_Docs/wiki/REST_api_reference',\n            },\n        });\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class huobipro extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'huobipro',\n            'name': 'Huobi Pro',\n            'countries': 'CN',\n            'rateLimit': 2000,\n            'userAgent': this.userAgents['chrome39'],\n            'version': 'v1',\n            'accounts': undefined,\n            'accountsById': undefined,\n            'hostname': 'api.huobi.pro',\n            'hasCORS': false,\n            // obsolete metainfo structure\n            'hasFetchOHLCV': true,\n            'hasFetchOrders': true,\n            'hasFetchOpenOrders': true,\n            // new metainfo structure\n            'has': {\n                'fetchOHCLV': true,\n                'fetchOrders': true,\n                'fetchOpenOrders': true,\n            },\n            'timeframes': {\n                '1m': '1min',\n                '5m': '5min',\n                '15m': '15min',\n                '30m': '30min',\n                '1h': '60min',\n                '1d': '1day',\n                '1w': '1week',\n                '1M': '1mon',\n                '1y': '1year',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766569-15aa7b9a-5edd-11e7-9e7f-44791f4ee49c.jpg',\n                'api': 'https://api.huobi.pro',\n                'www': 'https://www.huobi.pro',\n                'doc': 'https://github.com/huobiapi/API_Docs/wiki/REST_api_reference',\n            },\n            'api': {\n                'market': {\n                    'get': [\n                        'history/kline', // 获取K线数据\n                        'detail/merged', // 获取聚合行情(Ticker)\n                        'depth', // 获取 Market Depth 数据\n                        'trade', // 获取 Trade Detail 数据\n                        'history/trade', // 批量获取最近的交易记录\n                        'detail', // 获取 Market Detail 24小时成交量数据\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'common/symbols', // 查询系统支持的所有交易对\n                        'common/currencys', // 查询系统支持的所有币种\n                        'common/timestamp', // 查询系统当前时间\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'account/accounts', // 查询当前用户的所有账户(即account-id)\n                        'account/accounts/{id}/balance', // 查询指定账户的余额\n                        'order/orders/{id}', // 查询某个订单详情\n                        'order/orders/{id}/matchresults', // 查询某个订单的成交明细\n                        'order/orders', // 查询当前委托、历史委托\n                        'order/matchresults', // 查询当前成交、历史成交\n                        'dw/withdraw-virtual/addresses', // 查询虚拟币提现地址\n                    ],\n                    'post': [\n                        'order/orders/place', // 创建并执行一个新订单 (一步下单， 推荐使用)\n                        'order/orders', // 创建一个新的订单请求 （仅创建订单，不执行下单）\n                        'order/orders/{id}/place', // 执行一个订单 （仅执行已创建的订单）\n                        'order/orders/{id}/submitcancel', // 申请撤销一个订单请求\n                        'order/orders/batchcancel', // 批量撤销订单\n                        'dw/balance/transfer', // 资产划转\n                        'dw/withdraw-virtual/create', // 申请提现虚拟币\n                        'dw/withdraw-virtual/{id}/place', // 确认申请虚拟币提现\n                        'dw/withdraw-virtual/{id}/cancel', // 申请取消提现虚拟币\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicGetCommonSymbols ();\n        let markets = response['data'];\n        let numMarkets = markets.length;\n        if (numMarkets < 1)\n            throw new ExchangeError (this.id + ' publicGetCommonSymbols returned empty response: ' + this.json (response));\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let baseId = market['base-currency'];\n            let quoteId = market['quote-currency'];\n            let base = baseId.toUpperCase ();\n            let quote = quoteId.toUpperCase ();\n            let id = baseId + quoteId;\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            let precision = {\n                'amount': market['amount-precision'],\n                'price': market['price-precision'],\n            };\n            let lot = Math.pow (10, -precision['amount']);\n            let maker = (base == 'OMG') ? 0 : 0.2 / 100;\n            let taker = (base == 'OMG') ? 0 : 0.2 / 100;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'lot': lot,\n                'precision': precision,\n                'taker': taker,\n                'maker': maker,\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': Math.pow (10, precision['amount']),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision['price']),\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': 0,\n                        'max': undefined,\n                    },\n                },\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let last = undefined;\n        if ('last' in ticker)\n            last = ticker['last'];\n        let timestamp = this.milliseconds ();\n        if ('ts' in ticker)\n            timestamp = ticker['ts'];\n        let bid = undefined;\n        let ask = undefined;\n        if ('bid' in ticker) {\n            if (ticker['bid'])\n                bid = this.safeFloat (ticker['bid'], 0);\n        }\n        if ('ask' in ticker) {\n            if (ticker['ask'])\n                ask = this.safeFloat (ticker['ask'], 0);\n        }\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': ticker['high'],\n            'low': ticker['low'],\n            'bid': bid,\n            'ask': ask,\n            'vwap': undefined,\n            'open': ticker['open'],\n            'close': ticker['close'],\n            'first': undefined,\n            'last': last,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['amount']),\n            'quoteVolume': ticker['vol'],\n            'info': ticker,\n        };\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.marketGetDepth (this.extend ({\n            'symbol': market['id'],\n            'type': 'step0',\n        }, params));\n        if ('tick' in response) {\n            if (!response['tick']) {\n                throw new ExchangeError (this.id + ' fetchOrderBook() returned empty response: ' + this.json (response));\n            }\n            return this.parseOrderBook (response['tick'], response['tick']['ts']);\n        }\n        throw new ExchangeError (this.id + ' fetchOrderBook() returned unrecognized response: ' + this.json (response));\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.marketGetDetailMerged (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTicker (response['tick'], market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['ts'];\n        return {\n            'info': trade,\n            'id': trade['id'].toString (),\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['direction'],\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    parseTradesData (data, market, since = undefined, limit = undefined) {\n        let result = [];\n        for (let i = 0; i < data.length; i++) {\n            let trades = this.parseTrades (data[i]['data'], market, since, limit);\n            for (let k = 0; k < trades.length; k++) {\n                result.push (trades[k]);\n            }\n        }\n        return result;\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.marketGetHistoryTrade (this.extend ({\n            'symbol': market['id'],\n            'size': 2000,\n        }, params));\n        return this.parseTradesData (response['data'], market, since, limit);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv['id'] * 1000,\n            ohlcv['open'],\n            ohlcv['high'],\n            ohlcv['low'],\n            ohlcv['close'],\n            ohlcv['vol'],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.marketGetHistoryKline (this.extend ({\n            'symbol': market['id'],\n            'period': this.timeframes[timeframe],\n            'size': 2000, // max = 2000\n        }, params));\n        return this.parseOHLCVs (response['data'], market, timeframe, since, limit);\n    }\n\n    async loadAccounts (reload = false) {\n        if (reload) {\n            this.accounts = await this.fetchAccounts ();\n        } else {\n            if (this.accounts) {\n                return this.accounts;\n            } else {\n                this.accounts = await this.fetchAccounts ();\n                this.accountsById = this.indexBy (this.accounts, 'id');\n            }\n        }\n        return this.accounts;\n    }\n\n    async fetchAccounts () {\n        await this.loadMarkets ();\n        let response = await this.privateGetAccountAccounts ();\n        return response['data'];\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        await this.loadAccounts ();\n        let response = await this.privateGetAccountAccountsIdBalance (this.extend ({\n            'id': this.accounts[0]['id'],\n        }, params));\n        let balances = response['data']['list'];\n        let result = { 'info': response };\n        for (let i = 0; i < balances.length; i++) {\n            let balance = balances[i];\n            let uppercase = balance['currency'].toUpperCase ();\n            let currency = this.commonCurrencyCode (uppercase);\n            let account = undefined;\n            if (currency in result)\n                account = result[currency];\n            else\n                account = this.account ();\n            if (balance['type'] == 'trade')\n                account['free'] = parseFloat (balance['balance']);\n            if (balance['type'] == 'frozen')\n                account['used'] = parseFloat (balance['balance']);\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrders() requires a symbol parameter');\n        this.load_markets ();\n        let market = this.market (symbol);\n        let status = undefined;\n        if ('type' in params) {\n            status = params['type'];\n        } else if ('status' in params) {\n            status = params['status'];\n        } else {\n            throw new ExchangeError (this.id + ' fetchOrders() requires type param or status param for spot market ' + symbol + '(0 or \"open\" for unfilled or partial filled orders, 1 or \"closed\" for filled orders)');\n        }\n        if ((status == 0) || (status == 'open')) {\n            status = 'submitted,partial-filled';\n        } else if ((status == 1) || (status == 'closed')) {\n            status = 'filled,partial-canceled';\n        } else {\n            throw new ExchangeError (this.id + ' fetchOrders() wrong type param or status param for spot market ' + symbol + '(0 or \"open\" for unfilled or partial filled orders, 1 or \"closed\" for filled orders)');\n        }\n        let response = await this.privateGetOrderOrders (this.extend ({\n            'symbol': market['id'],\n            'states': status,\n        }));\n        return this.parseOrders (response['data'], market, since, limit);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let open = 0; // 0 for unfilled orders, 1 for filled orders\n        return this.fetchOrders (symbol, undefined, undefined, this.extend ({\n            'status': open,\n        }, params));\n    }\n\n    parseOrderStatus (status) {\n        if (status == 'partial-filled') {\n            return 'open';\n        } else if (status == 'filled') {\n            return 'closed';\n        } else if (status == 'canceled') {\n            return 'canceled';\n        } else if (status == 'submitted') {\n            return 'open';\n        }\n        return status;\n    }\n\n    parseOrder (order, market = undefined) {\n        let side = undefined;\n        let type = undefined;\n        let status = undefined;\n        if ('type' in order) {\n            let orderType = order['type'].split ('-');\n            side = orderType[0];\n            type = orderType[1];\n            status = this.parseOrderStatus (order['state']);\n        }\n        let symbol = undefined;\n        if (!market) {\n            if ('symbol' in order) {\n                if (order['symbol'] in this.markets_by_id) {\n                    let marketId = order['symbol'];\n                    market = this.markets_by_id[marketId];\n                }\n            }\n        }\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = order['created-at'];\n        let amount = parseFloat (order['amount']);\n        let filled = parseFloat (order['field-amount']);\n        let remaining = amount - filled;\n        let price = parseFloat (order['price']);\n        let cost = parseFloat (order['field-cash-amount']);\n        let average = 0;\n        if (filled)\n            average = parseFloat (cost / filled);\n        let result = {\n            'info': order,\n            'id': order['id'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': price,\n            'average': average,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': undefined,\n        };\n        return result;\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        await this.loadAccounts ();\n        let market = this.market (symbol);\n        let order = {\n            'account-id': this.accounts[0]['id'],\n            'amount': this.amountToPrecision (symbol, amount),\n            'symbol': market['id'],\n            'type': side + '-' + type,\n        };\n        if (type == 'limit')\n            order['price'] = this.priceToPrecision (symbol, price);\n        let response = await this.privatePostOrderOrdersPlace (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['data'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostOrderOrdersIdSubmitcancel ({ 'id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = '/';\n        if (api == 'market')\n            url += api;\n        else\n            url += this.version;\n        url += '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            let timestamp = this.YmdHMS (this.milliseconds (), 'T');\n            let request = this.keysort (this.extend ({\n                'SignatureMethod': 'HmacSHA256',\n                'SignatureVersion': '2',\n                'AccessKeyId': this.apiKey,\n                'Timestamp': timestamp,\n            }, query));\n            let auth = this.urlencode (request);\n            let payload = [ method, this.hostname, url, auth ].join (\"\\n\");\n            let signature = this.hmac (this.encode (payload), this.encode (this.secret), 'sha256', 'base64');\n            auth += '&' + this.urlencode ({ 'Signature': signature });\n            url += '?' + auth;\n            if (method == 'POST') {\n                body = this.json (query);\n                headers = {\n                    'Content-Type': 'application/json',\n                };\n            }\n        } else {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        }\n        url = this.urls['api'] + url;\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('status' in response)\n            if (response['status'] == 'error')\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class independentreserve extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'independentreserve',\n            'name': 'Independent Reserve',\n            'countries': [ 'AU', 'NZ' ], // Australia, New Zealand\n            'rateLimit': 1000,\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30521662-cf3f477c-9bcb-11e7-89bc-d1ac85012eda.jpg',\n                'api': {\n                    'public': 'https://api.independentreserve.com/Public',\n                    'private': 'https://api.independentreserve.com/Private',\n                },\n                'www': 'https://www.independentreserve.com',\n                'doc': 'https://www.independentreserve.com/API',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'GetValidPrimaryCurrencyCodes',\n                        'GetValidSecondaryCurrencyCodes',\n                        'GetValidLimitOrderTypes',\n                        'GetValidMarketOrderTypes',\n                        'GetValidOrderTypes',\n                        'GetValidTransactionTypes',\n                        'GetMarketSummary',\n                        'GetOrderBook',\n                        'GetTradeHistorySummary',\n                        'GetRecentTrades',\n                        'GetFxRates',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'PlaceLimitOrder',\n                        'PlaceMarketOrder',\n                        'CancelOrder',\n                        'GetOpenOrders',\n                        'GetClosedOrders',\n                        'GetClosedFilledOrders',\n                        'GetOrderDetails',\n                        'GetAccounts',\n                        'GetTransactions',\n                        'GetDigitalCurrencyDepositAddress',\n                        'GetDigitalCurrencyDepositAddresses',\n                        'SynchDigitalCurrencyDepositAddressWithBlockchain',\n                        'WithdrawDigitalCurrency',\n                        'RequestFiatWithdrawal',\n                        'GetTrades',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let baseCurrencies = await this.publicGetValidPrimaryCurrencyCodes ();\n        let quoteCurrencies = await this.publicGetValidSecondaryCurrencyCodes ();\n        let result = [];\n        for (let i = 0; i < baseCurrencies.length; i++) {\n            let baseId = baseCurrencies[i];\n            let baseIdUppercase = baseId.toUpperCase ();\n            let base = this.commonCurrencyCode (baseIdUppercase);\n            for (let j = 0; j < quoteCurrencies.length; j++) {\n                let quoteId = quoteCurrencies[j];\n                let quoteIdUppercase = quoteId.toUpperCase ();\n                let quote = this.commonCurrencyCode (quoteIdUppercase);\n                let id = baseId + '/' + quoteId;\n                let symbol = base + '/' + quote;\n                let taker = 0.5 / 100;\n                let maker = 0.5 / 100;\n                result.push ({\n                    'id': id,\n                    'symbol': symbol,\n                    'base': base,\n                    'quote': quote,\n                    'baseId': baseId,\n                    'quoteId': quoteId,\n                    'taker': taker,\n                    'maker': maker,\n                    'info': id,\n                });\n            }\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privatePostGetAccounts ();\n        let result = { 'info': balances };\n        for (let i = 0; i < balances.length; i++) {\n            let balance = balances[i];\n            let currencyCode = balance['CurrencyCode'];\n            let uppercase = currencyCode.toUpperCase ();\n            let currency = this.commonCurrencyCode (uppercase);\n            let account = this.account ();\n            account['free'] = balance['AvailableBalance'];\n            account['total'] = balance['TotalBalance'];\n            account['used'] = account['total'] - account['free'];\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetOrderBook (this.extend ({\n            'primaryCurrencyCode': market['baseId'],\n            'secondaryCurrencyCode': market['quoteId'],\n        }, params));\n        let timestamp = this.parse8601 (response['CreatedTimestampUtc']);\n        return this.parseOrderBook (response, timestamp, 'BuyOrders', 'SellOrders', 'Price', 'Volume');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.parse8601 (ticker['CreatedTimestampUtc']);\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': ticker['DayHighestPrice'],\n            'low': ticker['DayLowestPrice'],\n            'bid': ticker['CurrentHighestBidPrice'],\n            'ask': ticker['CurrentLowestOfferPrice'],\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': ticker['LastPrice'],\n            'change': undefined,\n            'percentage': undefined,\n            'average': ticker['DayAvgPrice'],\n            'baseVolume': ticker['DayVolumeXbtInSecondaryCurrrency'],\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetMarketSummary (this.extend ({\n            'primaryCurrencyCode': market['baseId'],\n            'secondaryCurrencyCode': market['quoteId'],\n        }, params));\n        return this.parseTicker (response, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = this.parse8601 (trade['TradeTimestampUtc']);\n        return {\n            'id': undefined,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': undefined,\n            'type': undefined,\n            'side': undefined,\n            'price': trade['SecondaryCurrencyTradePrice'],\n            'amount': trade['PrimaryCurrencyAmount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetRecentTrades (this.extend ({\n            'primaryCurrencyCode': market['baseId'],\n            'secondaryCurrencyCode': market['quoteId'],\n            'numberOfRecentTradesToRetrieve': 50, // max = 50\n        }, params));\n        return this.parseTrades (response['Trades'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let capitalizedOrderType = this.capitalize (type);\n        let method = 'privatePostPlace' + capitalizedOrderType + 'Order';\n        let orderType = capitalizedOrderType;\n        orderType += (side == 'sell') ?  'Offer' : 'Bid';\n        let order = this.ordered ({\n            'primaryCurrencyCode': market['baseId'],\n            'secondaryCurrencyCode': market['quoteId'],\n            'orderType': orderType,\n        });\n        if (type == 'limit')\n            order['price'] = price;\n        order['volume'] = amount;\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['OrderGuid'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelOrder ({ 'orderGuid': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api] + '/' + path;\n        if (api == 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let auth = [\n                url,\n                'apiKey=' + this.apiKey,\n                'nonce=' + nonce.toString (),\n            ];\n            let keysorted = this.keysort (params);\n            let keys = Object.keys (keysorted);\n            for (let i = 0; i < keys.length; i++) {\n                let key = keys[i];\n                auth.push (key + '=' + params[key]);\n            }\n            let message = auth.join (',');\n            let signature = this.hmac (this.encode (message), this.encode (this.secret));\n            let query = this.keysort (this.extend ({\n                'apiKey': this.apiKey,\n                'nonce': nonce,\n                'signature': signature,\n            }, params));\n            body = this.json (query);\n            headers = { 'Content-Type': 'application/json' };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        // todo error handling\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class itbit extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'itbit',\n            'name': 'itBit',\n            'countries': 'US',\n            'rateLimit': 2000,\n            'version': 'v1',\n            'hasCORS': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27822159-66153620-60ad-11e7-89e7-005f6d7f3de0.jpg',\n                'api': 'https://api.itbit.com',\n                'www': 'https://www.itbit.com',\n                'doc': [\n                    'https://api.itbit.com/docs',\n                    'https://www.itbit.com/api',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'markets/{symbol}/ticker',\n                        'markets/{symbol}/order_book',\n                        'markets/{symbol}/trades',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'wallets',\n                        'wallets/{walletId}',\n                        'wallets/{walletId}/balances/{currencyCode}',\n                        'wallets/{walletId}/funding_history',\n                        'wallets/{walletId}/trades',\n                        'wallets/{walletId}/orders/{id}',\n                    ],\n                    'post': [\n                        'wallet_transfers',\n                        'wallets',\n                        'wallets/{walletId}/cryptocurrency_deposits',\n                        'wallets/{walletId}/cryptocurrency_withdrawals',\n                        'wallets/{walletId}/orders',\n                        'wire_withdrawal',\n                    ],\n                    'delete': [\n                        'wallets/{walletId}/orders/{id}',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/USD': { 'id': 'XBTUSD', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD' },\n                'BTC/SGD': { 'id': 'XBTSGD', 'symbol': 'BTC/SGD', 'base': 'BTC', 'quote': 'SGD' },\n                'BTC/EUR': { 'id': 'XBTEUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR' },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0,\n                    'taker': 0.2 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetMarketsSymbolOrderBook (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetMarketsSymbolTicker (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        let serverTimeUTC = ('serverTimeUTC' in ticker);\n        if (!serverTimeUTC)\n            throw new ExchangeError (this.id + ' fetchTicker returned a bad response: ' + this.json (ticker));\n        let timestamp = this.parse8601 (ticker['serverTimeUTC']);\n        let vwap = parseFloat (ticker['vwap24h']);\n        let baseVolume = parseFloat (ticker['volume24h']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high24h']),\n            'low': parseFloat (ticker['low24h']),\n            'bid': this.safeFloat (ticker, 'bid'),\n            'ask': this.safeFloat (ticker, 'ask'),\n            'vwap': vwap,\n            'open': parseFloat (ticker['openToday']),\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['lastPrice']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = this.parse8601 (trade['timestamp']);\n        let id = trade['matchNumber'].toString ();\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': id,\n            'order': id,\n            'type': undefined,\n            'side': undefined,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetMarketsSymbolTrades (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTrades (response['recentTrades'], market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privateGetBalances ();\n        let balances = response['balances'];\n        let result = { 'info': response };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let account = {\n                'free': parseFloat (balance['availableBalance']),\n                'used': 0.0,\n                'total': parseFloat (balance['totalBalance']),\n            };\n            account['used'] = account['total'] - account['free'];\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    fetchWallets () {\n        return this.privateGetWallets ();\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type == 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let walletIdInParams = ('walletId' in params);\n        if (!walletIdInParams)\n            throw new ExchangeError (this.id + ' createOrder requires a walletId parameter');\n        amount = amount.toString ();\n        price = price.toString ();\n        let market = this.market (symbol);\n        let order = {\n            'side': side,\n            'type': type,\n            'currency': market['base'],\n            'amount': amount,\n            'display': amount,\n            'price': price,\n            'instrument': market['id'],\n        };\n        let response = await this.privatePostTradeAdd (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        let walletIdInParams = ('walletId' in params);\n        if (!walletIdInParams)\n            throw new ExchangeError (this.id + ' cancelOrder requires a walletId parameter');\n        return await this.privateDeleteWalletsWalletIdOrdersId (this.extend ({\n            'id': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            if (Object.keys (query).length)\n                body = this.json (query);\n            else\n                body = '';\n            let nonce = this.nonce ().toString ();\n            let timestamp = nonce;\n            let auth = [ method, url, body, nonce, timestamp ];\n            let message = nonce + this.json (auth);\n            let hash = this.hash (this.encode (message), 'sha256', 'binary');\n            let binhash = this.binaryConcat (url, hash);\n            let signature = this.hmac (binhash, this.encode (this.secret), 'sha512', 'base64');\n            headers = {\n                'Authorization': this.apiKey + ':' + signature,\n                'Content-Type': 'application/json',\n                'X-Auth-Timestamp': timestamp,\n                'X-Auth-Nonce': nonce,\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('code' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst btcbox = require ('./btcbox.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class jubi extends btcbox {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'jubi',\n            'name': 'jubi.com',\n            'countries': 'CN',\n            'rateLimit': 1500,\n            'version': 'v1',\n            'hasCORS': false,\n            'hasFetchTickers': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766581-9d397d9a-5edd-11e7-8fb9-5d8236c0e692.jpg',\n                'api': 'https://www.jubi.com/api',\n                'www': 'https://www.jubi.com',\n                'doc': 'https://www.jubi.com/help/api.html',\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetAllticker ();\n        let keys = Object.keys (markets);\n        let result = [];\n        for (let p = 0; p < keys.length; p++) {\n            let id = keys[p];\n            let base = id.toUpperCase ();\n            let quote = 'CNY'; // todo\n            let symbol = base + '/' + quote;\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': id,\n            });\n        }\n        return result;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeNotAvailable, ExchangeError, OrderNotFound, DDoSProtection, InvalidNonce, InsufficientFunds, CancelPending, InvalidOrder } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class kraken extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'kraken',\n            'name': 'Kraken',\n            'countries': 'US',\n            'version': '0',\n            'rateLimit': 3000,\n            'hasCORS': false,\n            // obsolete metainfo interface\n            'hasFetchTickers': true,\n            'hasFetchOHLCV': true,\n            'hasFetchOrder': true,\n            'hasFetchOpenOrders': true,\n            'hasFetchClosedOrders': true,\n            'hasFetchMyTrades': true,\n            'hasWithdraw': true,\n            'hasFetchCurrencies': true,\n            // new metainfo interface\n            'has': {\n                'fetchCurrencies': true,\n                'fetchTickers': true,\n                'fetchOHLCV': true,\n                'fetchOrder': true,\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': true,\n                'fetchMyTrades': true,\n                'withdraw': true,\n            },\n            'marketsByAltname': {},\n            'timeframes': {\n                '1m': '1',\n                '5m': '5',\n                '15m': '15',\n                '30m': '30',\n                '1h': '60',\n                '4h': '240',\n                '1d': '1440',\n                '1w': '10080',\n                '2w': '21600',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766599-22709304-5ede-11e7-9de1-9f33732e1509.jpg',\n                'api': 'https://api.kraken.com',\n                'www': 'https://www.kraken.com',\n                'doc': [\n                    'https://www.kraken.com/en-us/help/api',\n                    'https://github.com/nothingisdead/npm-kraken-api',\n                ],\n                'fees': [\n                    'https://www.kraken.com/en-us/help/fees',\n                    'https://support.kraken.com/hc/en-us/articles/201396777-What-are-the-deposit-fees-',\n                    'https://support.kraken.com/hc/en-us/articles/201893608-What-are-the-withdrawal-fees-',\n                ],\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': true,\n                    'percentage': true,\n                    'taker': 0.26 / 100,\n                    'maker': 0.16 / 100,\n                    'tiers': {\n                        'taker': [\n                            [0, 0.26 / 100],\n                            [50000, 0.24 / 100],\n                            [100000, 0.22 / 100],\n                            [250000, 0.2 / 100],\n                            [500000, 0.18 / 100],\n                            [1000000, 0.16 / 100],\n                            [2500000, 0.14 / 100],\n                            [5000000, 0.12 / 100],\n                            [10000000, 0.1 / 100],\n                        ],\n                        'maker': [\n                            [0, 0.16 / 100],\n                            [50000, 0.14 / 100],\n                            [100000, 0.12 / 100],\n                            [250000, 0.10 / 100],\n                            [500000, 0.8 / 100],\n                            [1000000, 0.6 / 100],\n                            [2500000, 0.4 / 100],\n                            [5000000, 0.2 / 100],\n                            [10000000, 0.0 / 100],\n                        ],\n                    },\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.001,\n                        'ETH': 0.005,\n                        'XRP': 0.02,\n                        'XLM': 0.00002,\n                        'LTC': 0.02,\n                        'DOGE': 2,\n                        'ZEC': 0.00010,\n                        'ICN': 0.02,\n                        'REP': 0.01,\n                        'ETC': 0.005,\n                        'MLN': 0.003,\n                        'XMR': 0.05,\n                        'DASH': 0.005,\n                        'GNO': 0.01,\n                        'EOS': 0.5,\n                        'BCH': 0.001,\n                        'USD': 5, // if domestic wire\n                        'EUR': 5, // if domestic wire\n                        'CAD': 10, // CAD EFT Withdrawal\n                        'JPY': 300, // if domestic wire\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                        'ETH': 0,\n                        'XRP': 0,\n                        'XLM': 0,\n                        'LTC': 0,\n                        'DOGE': 0,\n                        'ZEC': 0,\n                        'ICN': 0,\n                        'REP': 0,\n                        'ETC': 0,\n                        'MLN': 0,\n                        'XMR': 0,\n                        'DASH': 0,\n                        'GNO': 0,\n                        'EOS': 0,\n                        'BCH': 0,\n                        'USD': 5, // if domestic wire\n                        'EUR': 0, // free deposit if EUR SEPA Deposit\n                        'CAD': 5, // if domestic wire\n                        'JPY': 0, // Domestic Deposit (Free, ¥5,000 deposit minimum)\n                    },\n                },\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'Assets',\n                        'AssetPairs',\n                        'Depth',\n                        'OHLC',\n                        'Spread',\n                        'Ticker',\n                        'Time',\n                        'Trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'AddOrder',\n                        'Balance',\n                        'CancelOrder',\n                        'ClosedOrders',\n                        'DepositAddresses',\n                        'DepositMethods',\n                        'DepositStatus',\n                        'Ledgers',\n                        'OpenOrders',\n                        'OpenPositions',\n                        'QueryLedgers',\n                        'QueryOrders',\n                        'QueryTrades',\n                        'TradeBalance',\n                        'TradesHistory',\n                        'TradeVolume',\n                        'Withdraw',\n                        'WithdrawCancel',\n                        'WithdrawInfo',\n                        'WithdrawStatus',\n                    ],\n                },\n            },\n        });\n    }\n\n    costToPrecision (symbol, cost) {\n        return this.truncate (parseFloat (cost), this.markets[symbol]['precision']['price']);\n    }\n\n    feeToPrecision (symbol, fee) {\n        return this.truncate (parseFloat (fee), this.markets[symbol]['precision']['amount']);\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (body.indexOf ('Invalid order') >= 0)\n            throw new InvalidOrder (this.id + ' ' + body);\n        if (body.indexOf ('Invalid nonce') >= 0)\n            throw new InvalidNonce (this.id + ' ' + body);\n        if (body.indexOf ('Insufficient funds') >= 0)\n            throw new InsufficientFunds (this.id + ' ' + body);\n        if (body.indexOf ('Cancel pending') >= 0)\n            throw new CancelPending (this.id + ' ' + body);\n        if (body.indexOf ('Invalid arguments:volume') >= 0)\n            throw new InvalidOrder (this.id + ' ' + body);\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetAssetPairs ();\n        let keys = Object.keys (markets['result']);\n        let result = [];\n        for (let i = 0; i < keys.length; i++) {\n            let id = keys[i];\n            let market = markets['result'][id];\n            let base = market['base'];\n            let quote = market['quote'];\n            if ((base[0] === 'X') || (base[0] === 'Z'))\n                base = base.slice (1);\n            if ((quote[0] === 'X') || (quote[0] === 'Z'))\n                quote = quote.slice (1);\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let darkpool = id.indexOf ('.d') >= 0;\n            let symbol = darkpool ? market['altname'] : (base + '/' + quote);\n            let maker = undefined;\n            if ('fees_maker' in market) {\n                maker = parseFloat (market['fees_maker'][0][1]) / 100;\n            }\n            let precision = {\n                'amount': market['lot_decimals'],\n                'price': market['pair_decimals'],\n            };\n            let lot = Math.pow (10, -precision['amount']);\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'darkpool': darkpool,\n                'info': market,\n                'altname': market['altname'],\n                'maker': maker,\n                'taker': parseFloat (market['fees'][0][1]) / 100,\n                'lot': lot,\n                'active': true,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': Math.pow (10, precision['amount']),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision['price']),\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': 0,\n                        'max': undefined,\n                    },\n                },\n            });\n        }\n        result = this.appendInactiveMarkets (result);\n        this.marketsByAltname = this.indexBy (result, 'altname');\n        return result;\n    }\n\n    appendInactiveMarkets (result = []) {\n        let precision = { 'amount': 8, 'price': 8 };\n        let costLimits = { 'min': 0, 'max': undefined };\n        let priceLimits = { 'min': Math.pow (10, -precision['price']), 'max': undefined };\n        let amountLimits = { 'min': Math.pow (10, -precision['amount']), 'max': Math.pow (10, precision['amount']) };\n        let limits = { 'amount': amountLimits, 'price': priceLimits, 'cost': costLimits };\n        let defaults = {\n            'darkpool': false,\n            'info': undefined,\n            'maker': undefined,\n            'taker': undefined,\n            'lot': amountLimits['min'],\n            'active': false,\n            'precision': precision,\n            'limits': limits,\n        };\n        let markets = [\n            { 'id': 'XXLMZEUR', 'symbol': 'XLM/EUR', 'base': 'XLM', 'quote': 'EUR', 'altname': 'XLMEUR' },\n        ];\n        for (let i = 0; i < markets.length; i++) {\n            result.push (this.extend (defaults, markets[i]));\n        }\n        return result;\n    }\n\n    async fetchCurrencies (params = {}) {\n        let response = await this.publicGetAssets (params);\n        let currencies = response['result'];\n        let ids = Object.keys (currencies);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let currency = currencies[id];\n            // todo: will need to rethink the fees\n            // to add support for multiple withdrawal/deposit methods and\n            // differentiated fees for each particular method\n            let code = this.commonCurrencyCode (currency['altname']);\n            let precision = currency['decimals'];\n            result[code] = {\n                'id': id,\n                'code': code,\n                'info': currency,\n                'name': code,\n                'active': true,\n                'status': 'ok',\n                'fee': undefined,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': undefined,\n                        'max': Math.pow (10, precision),\n                    },\n                },\n            };\n        }\n        return result;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let darkpool = symbol.indexOf ('.d') >= 0;\n        if (darkpool)\n            throw new ExchangeError (this.id + ' does not provide an order book for darkpool symbol ' + symbol);\n        let market = this.market (symbol);\n        let response = await this.publicGetDepth (this.extend ({\n            'pair': market['id'],\n            // 'count': 100,\n        }, params));\n        let orderbook = response['result'][market['id']];\n        return this.parseOrderBook (orderbook);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let baseVolume = parseFloat (ticker['v'][1]);\n        let vwap = parseFloat (ticker['p'][1]);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['h'][1]),\n            'low': parseFloat (ticker['l'][1]),\n            'bid': parseFloat (ticker['b'][0]),\n            'ask': parseFloat (ticker['a'][0]),\n            'vwap': vwap,\n            'open': parseFloat (ticker['o']),\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['c'][0]),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let pairs = [];\n        for (let s = 0; s < this.symbols.length; s++) {\n            let symbol = this.symbols[s];\n            let market = this.markets[symbol];\n            if (market['active'])\n                if (!market['darkpool'])\n                    pairs.push (market['id']);\n        }\n        let filter = pairs.join (',');\n        let response = await this.publicGetTicker (this.extend ({\n            'pair': filter,\n        }, params));\n        let tickers = response['result'];\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let darkpool = symbol.indexOf ('.d') >= 0;\n        if (darkpool)\n            throw new ExchangeError (this.id + ' does not provide a ticker for darkpool symbol ' + symbol);\n        let market = this.market (symbol);\n        let response = await this.publicGetTicker (this.extend ({\n            'pair': market['id'],\n        }, params));\n        let ticker = response['result'][market['id']];\n        return this.parseTicker (ticker, market);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv[0] * 1000,\n            parseFloat (ohlcv[1]),\n            parseFloat (ohlcv[2]),\n            parseFloat (ohlcv[3]),\n            parseFloat (ohlcv[4]),\n            parseFloat (ohlcv[6]),\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'pair': market['id'],\n            'interval': this.timeframes[timeframe],\n        };\n        if (since)\n            request['since'] = parseInt (since / 1000);\n        let response = await this.publicGetOHLC (this.extend (request, params));\n        let ohlcvs = response['result'][market['id']];\n        return this.parseOHLCVs (ohlcvs, market, timeframe, since, limit);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = undefined;\n        let side = undefined;\n        let type = undefined;\n        let price = undefined;\n        let amount = undefined;\n        let id = undefined;\n        let order = undefined;\n        let fee = undefined;\n        if (!market)\n            market = this.findMarketByAltnameOrId (trade['pair']);\n        if ('ordertxid' in trade) {\n            order = trade['ordertxid'];\n            id = trade['id'];\n            timestamp = parseInt (trade['time'] * 1000);\n            side = trade['type'];\n            type = trade['ordertype'];\n            price = parseFloat (trade['price']);\n            amount = parseFloat (trade['vol']);\n            if ('fee' in trade) {\n                let currency = undefined;\n                if (market)\n                    currency = market['quote'];\n                fee = {\n                    'cost': parseFloat (trade['fee']),\n                    'currency': currency,\n                };\n            }\n        } else {\n            timestamp = parseInt (trade[2] * 1000);\n            side = (trade[3] === 's') ? 'sell' : 'buy';\n            type = (trade[4] === 'l') ? 'limit' : 'market';\n            price = parseFloat (trade[0]);\n            amount = parseFloat (trade[1]);\n        }\n        let symbol = (market) ? market['symbol'] : undefined;\n        return {\n            'id': id,\n            'order': order,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': price,\n            'amount': amount,\n            'fee': fee,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let id = market['id'];\n        let response = await this.publicGetTrades (this.extend ({\n            'pair': id,\n        }, params));\n        let trades = response['result'][id];\n        return this.parseTrades (trades, market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostBalance ();\n        let balances = response['result'];\n        let result = { 'info': balances };\n        let currencies = Object.keys (balances);\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let code = currency;\n            // X-ISO4217-A3 standard currency codes\n            if (code[0] === 'X') {\n                code = code.slice (1);\n            } else if (code[0] === 'Z') {\n                code = code.slice (1);\n            }\n            code = this.commonCurrencyCode (code);\n            let balance = parseFloat (balances[currency]);\n            let account = {\n                'free': balance,\n                'used': 0.0,\n                'total': balance,\n            };\n            result[code] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let order = {\n            'pair': market['id'],\n            'type': side,\n            'ordertype': type,\n            'volume': this.amountToPrecision (symbol, amount),\n        };\n        if (type === 'limit')\n            order['price'] = this.priceToPrecision (symbol, price);\n        let response = await this.privatePostAddOrder (this.extend (order, params));\n        let length = response['result']['txid'].length;\n        let id = (length > 1) ? response['result']['txid'] : response['result']['txid'][0];\n        return {\n            'info': response,\n            'id': id,\n        };\n    }\n\n    findMarketByAltnameOrId (id) {\n        let result = undefined;\n        if (id in this.marketsByAltname) {\n            result = this.marketsByAltname[id];\n        } else if (id in this.markets_by_id) {\n            result = this.markets_by_id[id];\n        }\n        return result;\n    }\n\n    parseOrder (order, market = undefined) {\n        let description = order['descr'];\n        let side = description['type'];\n        let type = description['ordertype'];\n        let symbol = undefined;\n        if (!market)\n            market = this.findMarketByAltnameOrId (description['pair']);\n        let timestamp = parseInt (order['opentm'] * 1000);\n        let amount = parseFloat (order['vol']);\n        let filled = parseFloat (order['vol_exec']);\n        let remaining = amount - filled;\n        let fee = undefined;\n        let cost = this.safeFloat (order, 'cost');\n        let price = this.safeFloat (description, 'price');\n        if (!price)\n            price = this.safeFloat (order, 'price');\n        if (market) {\n            symbol = market['symbol'];\n            if ('fee' in order) {\n                let flags = order['oflags'];\n                let feeCost = this.safeFloat (order, 'fee');\n                fee = {\n                    'cost': feeCost,\n                    'rate': undefined,\n                };\n                if (flags.indexOf ('fciq') >= 0) {\n                    fee['currency'] = market['quote'];\n                } else if (flags.indexOf ('fcib') >= 0) {\n                    fee['currency'] = market['base'];\n                }\n            }\n        }\n        return {\n            'id': order['id'],\n            'info': order,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': order['status'],\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'fee': fee,\n            // 'trades': this.parseTrades (order['trades'], market),\n        };\n    }\n\n    parseOrders (orders, market = undefined, since = undefined, limit = undefined) {\n        let result = [];\n        let ids = Object.keys (orders);\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let order = this.extend ({ 'id': id }, orders[id]);\n            result.push (this.parseOrder (order, market));\n        }\n        return this.filterBySinceLimit (result, since, limit);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostQueryOrders (this.extend ({\n            'trades': true, // whether or not to include trades in output (optional, default false)\n            'txid': id, // comma delimited list of transaction ids to query info about (20 maximum)\n            // 'userref': 'optional', // restrict results to given user reference id (optional)\n        }, params));\n        let orders = response['result'];\n        let order = this.parseOrder (this.extend ({ 'id': id }, orders[id]));\n        return this.extend ({ 'info': response }, order);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            // 'type': 'all', // any position, closed position, closing position, no position\n            // 'trades': false, // whether or not to include trades related to position in output\n            // 'start': 1234567890, // starting unix timestamp or trade tx id of results (exclusive)\n            // 'end': 1234567890, // ending unix timestamp or trade tx id of results (inclusive)\n            // 'ofs' = result offset\n        };\n        if (since)\n            request['start'] = parseInt (since / 1000);\n        let response = await this.privatePostTradesHistory (this.extend (request, params));\n        let trades = response['result']['trades'];\n        let ids = Object.keys (trades);\n        for (let i = 0; i < ids.length; i++) {\n            trades[ids[i]]['id'] = ids[i];\n        }\n        return this.parseTrades (trades, undefined, since, limit);\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = undefined;\n        try {\n            response = await this.privatePostCancelOrder (this.extend ({\n                'txid': id,\n            }, params));\n        } catch (e) {\n            if (this.last_http_response)\n                if (this.last_http_response.indexOf ('EOrder:Unknown order') >= 0)\n                    throw new OrderNotFound (this.id + ' cancelOrder() error ' + this.last_http_response);\n            throw e;\n        }\n        return response;\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        if (since)\n            request['start'] = parseInt (since / 1000);\n        let response = await this.privatePostOpenOrders (this.extend (request, params));\n        let orders = this.parseOrders (response['result']['open'], undefined, since, limit);\n        return this.filterOrdersBySymbol (orders, symbol);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        if (since)\n            request['start'] = parseInt (since / 1000);\n        let response = await this.privatePostClosedOrders (this.extend (request, params));\n        let orders = this.parseOrders (response['result']['closed'], undefined, since, limit);\n        return this.filterOrdersBySymbol (orders, symbol);\n    }\n\n    async fetchDepositMethods (code = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        if (code) {\n            let currency = this.currency (code);\n            request['asset'] = currency['id'];\n        }\n        let response = await this.privatePostDepositMethods (this.extend (request, params));\n        return response['result'];\n    }\n\n    async createDepositAddress (currency, params = {}) {\n        let request = {\n            'new': 'true',\n        };\n        let response = await this.fetchDepositAddress (currency, this.extend (request, params));\n        return {\n            'currency': currency,\n            'address': response['address'],\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    async fetchDepositAddress (code, params = {}) {\n        let method = this.safeValue (params, 'method');\n        if (!method)\n            throw new ExchangeError (this.id + ' fetchDepositAddress() requires an extra `method` parameter');\n        await this.loadMarkets ();\n        let currency = this.currency (code);\n        let request = {\n            'asset': currency['id'],\n            'method': method,\n            'new': 'false',\n        };\n        let response = await this.privatePostDepositAddresses (this.extend (request, params));\n        let result = response['result'];\n        let numResults = result.length;\n        if (numResults < 1)\n            throw new ExchangeError (this.id + ' privatePostDepositAddresses() returned no addresses');\n        let address = this.safeString (result[0], 'address');\n        return {\n            'currency': code,\n            'address': address,\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        if ('key' in params) {\n            await this.loadMarkets ();\n            let response = await this.privatePostWithdraw (this.extend ({\n                'asset': currency,\n                'amount': amount,\n                // 'address': address, // they don't allow withdrawals to direct addresses\n            }, params));\n            return {\n                'info': response,\n                'id': response['result'],\n            };\n        }\n        throw new ExchangeError (this.id + \" withdraw requires a 'key' parameter (withdrawal key name, as set up on your account)\");\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = '/' + this.version + '/' + api + '/' + path;\n        if (api === 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            body = this.urlencode (this.extend ({ 'nonce': nonce }, params));\n            let auth = this.encode (nonce + body);\n            let hash = this.hash (auth, 'sha256', 'binary');\n            let binary = this.stringToBinary (this.encode (url));\n            let binhash = this.binaryConcat (binary, hash);\n            let secret = this.base64ToBinary (this.secret);\n            let signature = this.hmac (binhash, secret, 'sha512', 'base64');\n            headers = {\n                'API-Key': this.apiKey,\n                'API-Sign': this.decode (signature),\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        url = this.urls['api'] + url;\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response) {\n            let numErrors = response['error'].length;\n            if (numErrors) {\n                for (let i = 0; i < response['error'].length; i++) {\n                    if (response['error'][i] === 'EService:Unavailable')\n                        throw new ExchangeNotAvailable (this.id + ' ' + this.json (response));\n                    if (response['error'][i] === 'EService:Busy')\n                        throw new DDoSProtection (this.id + ' ' + this.json (response));\n                }\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n            }\n        }\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InvalidNonce, InvalidOrder, AuthenticationError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class kucoin extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'kucoin',\n            'name': 'Kucoin',\n            'countries': 'HK', // Hong Kong\n            'version': 'v1',\n            'rateLimit': 2000,\n            'hasCORS': false,\n            'userAgent': this.userAgents['chrome'],\n            // obsolete metainfo interface\n            'hasFetchTickers': true,\n            'hasFetchOHLCV': true,\n            'hasFetchOrder': false,\n            'hasFetchOrders': true,\n            'hasFetchClosedOrders': true,\n            'hasFetchOpenOrders': true,\n            'hasFetchMyTrades': false,\n            'hasFetchCurrencies': true,\n            'hasWithdraw': true,\n            // new metainfo interface\n            'has': {\n                'fetchTickers': true,\n                'fetchOHLCV': true, // see the method implementation below\n                'fetchOrder': false,\n                'fetchOrders': true,\n                'fetchClosedOrders': true,\n                'fetchOpenOrders': true,\n                'fetchMyTrades': false,\n                'fetchCurrencies': true,\n                'withdraw': true,\n            },\n            'timeframes': {\n                '1m': '1',\n                '5m': '5',\n                '15m': '15',\n                '30m': '30',\n                '1h': '60',\n                '8h': '480',\n                '1d': 'D',\n                '1w': 'W',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/33795655-b3c46e48-dcf6-11e7-8abe-dc4588ba7901.jpg',\n                'api': 'https://api.kucoin.com',\n                'www': 'https://kucoin.com',\n                'doc': 'https://kucoinapidocs.docs.apiary.io',\n                'fees': 'https://news.kucoin.com/en/fee',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'open/chart/config',\n                        'open/chart/history',\n                        'open/chart/symbol',\n                        'open/currencies',\n                        'open/deal-orders',\n                        'open/kline',\n                        'open/lang-list',\n                        'open/orders',\n                        'open/orders-buy',\n                        'open/orders-sell',\n                        'open/tick',\n                        'market/open/coin-info',\n                        'market/open/coins',\n                        'market/open/coins-trending',\n                        'market/open/symbols',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'account/balance',\n                        'account/{coin}/wallet/address',\n                        'account/{coin}/wallet/records',\n                        'account/{coin}/balance',\n                        'account/promotion/info',\n                        'account/promotion/sum',\n                        'deal-orders',\n                        'order/active',\n                        'order/active-map',\n                        'order/dealt',\n                        'referrer/descendant/count',\n                        'user/info',\n                    ],\n                    'post': [\n                        'account/{coin}/withdraw/apply',\n                        'account/{coin}/withdraw/cancel',\n                        'cancel-order',\n                        'order',\n                        'user/change-lang',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.0010,\n                    'taker': 0.0010,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'KCS': 2.0,\n                        'BTC': 0.0005,\n                        'USDT': 10.0,\n                        'ETH': 0.01,\n                        'LTC': 0.001,\n                        'NEO': 0.0,\n                        'GAS': 0.0,\n                        'KNC': 0.5,\n                        'BTM': 5.0,\n                        'QTUM': 0.1,\n                        'EOS': 0.5,\n                        'CVC': 3.0,\n                        'OMG': 0.1,\n                        'PAY': 0.5,\n                        'SNT': 20.0,\n                        'BHC': 1.0,\n                        'HSR': 0.01,\n                        'WTC': 0.1,\n                        'VEN': 2.0,\n                        'MTH': 10.0,\n                        'RPX': 1.0,\n                        'REQ': 20.0,\n                        'EVX': 0.5,\n                        'MOD': 0.5,\n                        'NEBL': 0.1,\n                        'DGB': 0.5,\n                        'CAG': 2.0,\n                        'CFD': 0.5,\n                        'RDN': 0.5,\n                        'UKG': 5.0,\n                        'BCPT': 5.0,\n                        'PPT': 0.1,\n                        'BCH': 0.0005,\n                        'STX': 2.0,\n                        'NULS': 1.0,\n                        'GVT': 0.1,\n                        'HST': 2.0,\n                        'PURA': 0.5,\n                        'SUB': 2.0,\n                        'QSP': 5.0,\n                        'POWR': 1.0,\n                        'FLIXX': 10.0,\n                        'LEND': 20.0,\n                        'AMB': 3.0,\n                        'RHOC': 2.0,\n                        'R': 2.0,\n                        'DENT': 50.0,\n                        'DRGN': 1.0,\n                        'ACT': 0.1,\n                    },\n                    'deposit': 0.00,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicGetMarketOpenSymbols ();\n        let markets = response['data'];\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let id = market['symbol'];\n            let base = market['coinType'];\n            let quote = market['coinTypePair'];\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            let precision = {\n                'amount': 8,\n                'price': 8,\n            };\n            let active = market['trading'];\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': active,\n                'info': market,\n                'lot': Math.pow (10, -precision['amount']),\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision['amount']),\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                },\n            }));\n        }\n        return result;\n    }\n\n    async fetchCurrencies (params = {}) {\n        let response = await this.publicGetMarketOpenCoins (params);\n        let currencies = response['data'];\n        let result = {};\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let id = currency['coin'];\n            // todo: will need to rethink the fees\n            // to add support for multiple withdrawal/deposit methods and\n            // differentiated fees for each particular method\n            let code = this.commonCurrencyCode (id);\n            let precision = currency['tradePrecision'];\n            let deposit = currency['enableDeposit'];\n            let withdraw = currency['enableWithdraw'];\n            let active = (deposit && withdraw);\n            result[code] = {\n                'id': id,\n                'code': code,\n                'info': currency,\n                'name': currency['name'],\n                'active': active,\n                'status': 'ok',\n                'fee': currency['withdrawFeeRate'], // todo: redesign\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': currency['withdrawMinAmount'],\n                        'max': Math.pow (10, precision),\n                    },\n                },\n            };\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetAccountBalance (this.extend ({\n            'limit': 20, // default 12, max 20\n            'page': 1,\n        }, params));\n        let balances = response['data'];\n        let result = { 'info': balances };\n        let indexed = this.indexBy (balances, 'coinType');\n        let keys = Object.keys (indexed);\n        for (let i = 0; i < keys.length; i++) {\n            let id = keys[i];\n            let currency = this.commonCurrencyCode (id);\n            let account = this.account ();\n            let balance = indexed[id];\n            let used = parseFloat (balance['freezeBalance']);\n            let free = parseFloat (balance['balance']);\n            let total = this.sum (free, used);\n            account['free'] = free;\n            account['used'] = used;\n            account['total'] = total;\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetOpenOrders (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        let orderbook = response['data'];\n        return this.parseOrderBook (orderbook, undefined, 'BUY', 'SELL');\n    }\n\n    parseOrder (order, market = undefined) {\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else {\n            symbol = order['coinType'] + '/' + order['coinTypePair'];\n        }\n        let timestamp = order['createdAt'];\n        let price = this.safeValue (order, 'price');\n        if (typeof price === 'undefined')\n            price = this.safeValue (order, 'dealPrice');\n        let amount = this.safeValue (order, 'amount');\n        let filled = this.safeValue (order, 'dealAmount', 0);\n        let remaining = this.safeValue (order, 'pendingAmount');\n        if (typeof amount === 'undefined')\n            if (typeof filled !== 'undefined')\n                if (typeof remaining !== 'undefined')\n                    amount = this.sum (filled, remaining);\n        let side = order['direction'].toLowerCase ();\n        let fee = undefined;\n        if ('fee' in order) {\n            fee = {\n                'cost': this.safeFloat (order, 'fee'),\n                'rate': this.safeFloat (order, 'feeRate'),\n            };\n            if (market)\n                fee['currency'] = market['base'];\n        }\n        let status = this.safeValue (order, 'status');\n        let result = {\n            'info': order,\n            'id': this.safeString (order, 'oid'),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': 'limit',\n            'side': side,\n            'price': price,\n            'amount': amount,\n            'cost': price * filled,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': fee,\n        };\n        return result;\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOpenOrders requires a symbol param');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n        };\n        let response = await this.privateGetOrderActiveMap (this.extend (request, params));\n        let orders = this.arrayConcat (response['data']['SELL'], response['data']['BUY']);\n        let result = [];\n        for (let i = 0; i < orders.length; i++) {\n            result.push (this.extend (orders[i], { 'status': 'open' }));\n        }\n        return this.parseOrders (result, market, since, limit);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let request = {};\n        await this.loadMarkets ();\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['symbol'] = market['id'];\n        }\n        if (since) {\n            request['since'] = since;\n        }\n        if (limit) {\n            request['limit'] = limit;\n        }\n        let response = await this.privateGetOrderDealt (this.extend (request, params));\n        let orders = response['data']['datas'];\n        let result = [];\n        for (let i = 0; i < orders.length; i++) {\n            result.push (this.extend (orders[i], { 'status': 'closed' }));\n        }\n        return this.parseOrders (result, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type !== 'limit')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let base = market['base'];\n        let order = {\n            'symbol': market['id'],\n            'type': side.toUpperCase (),\n            'price': this.priceToPrecision (symbol, price),\n            'amount': this.truncate (amount, this.currencies[base]['precision']),\n        };\n        let response = await this.privatePostOrder (this.extend (order, params));\n        return {\n            'info': response,\n            'id': this.safeString (response['data'], 'orderOid'),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' cancelOrder requires symbol argument');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n            'orderOid': id,\n        };\n        if ('type' in params) {\n            request['type'] = params['type'].toUpperCase ();\n        } else {\n            throw new ExchangeError (this.id + ' cancelOrder requires type (BUY or SELL) param');\n        }\n        let response = await this.privatePostCancelOrder (this.extend (request, params));\n        return response;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['datetime'];\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else {\n            symbol = ticker['coinType'] + '/' + ticker['coinTypePair'];\n        }\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'buy'),\n            'ask': this.safeFloat (ticker, 'sell'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'lastDealPrice'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'vol'),\n            'quoteVolume': this.safeFloat (ticker, 'volValue'),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        let response = await this.publicGetMarketOpenSymbols (params);\n        let tickers = response['data'];\n        let result = {};\n        for (let t = 0; t < tickers.length; t++) {\n            let ticker = this.parseTicker (tickers[t]);\n            let symbol = ticker['symbol'];\n            result[symbol] = ticker;\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetOpenTick (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        let ticker = response['data'];\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = trade[0];\n        let side = undefined;\n        if (trade[1] === 'BUY') {\n            side = 'buy';\n        } else if (trade[1] === 'SELL') {\n            side = 'sell';\n        }\n        return {\n            'id': undefined,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': 'limit',\n            'side': side,\n            'price': trade[2],\n            'amount': trade[3],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetOpenDealOrders (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTrades (response['data'], market, since, limit);\n    }\n\n    parseTradingViewOHLCVs (ohlcvs, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        let result = [];\n        for (let i = 0; i < ohlcvs['t'].length; i++) {\n            result.push ([\n                ohlcvs['t'][i],\n                ohlcvs['o'][i],\n                ohlcvs['h'][i],\n                ohlcvs['l'][i],\n                ohlcvs['c'][i],\n                ohlcvs['v'][i],\n            ]);\n        }\n        return this.parseOHLCVs (result, market, timeframe, since, limit);\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let end = this.seconds ();\n        let resolution = this.timeframes[timeframe];\n        // convert 'resolution' to minutes in order to calculate 'from' later\n        let minutes = resolution;\n        if (minutes === 'D') {\n            if (!limit)\n                limit = 30; // 30 days, 1 month\n            minutes = 1440;\n        } else if (minutes === 'W') {\n            if (!limit)\n                limit = 52; // 52 weeks, 1 year\n            minutes = 10080;\n        } else if (!limit) {\n            limit = 1440;\n            minutes = 1440;\n        }\n        let start = end - minutes * 60 * limit;\n        if (since) {\n            start = parseInt (since / 1000);\n            end = this.sum (start, minutes * 60 * limit);\n        }\n        let request = {\n            'symbol': market['id'],\n            'type': this.timeframes[timeframe],\n            'resolution': resolution,\n            'from': start,\n            'to': end,\n        };\n        let response = await this.publicGetOpenChartHistory (this.extend (request, params));\n        return this.parseTradingViewOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    async withdraw (code, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let currency = this.currency (code);\n        let response = await this.privatePostAccountCoinWithdrawApply (this.extend ({\n            'coin': currency['id'],\n            'amount': amount,\n            'address': address,\n        }, params));\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let endpoint = '/' + this.version + '/' + this.implodeParams (path, params);\n        let url = this.urls['api'] + endpoint;\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            // their nonce is always a calibrated synched milliseconds-timestamp\n            let nonce = this.milliseconds ();\n            let queryString = '';\n            nonce = nonce.toString ();\n            if (Object.keys (query).length) {\n                queryString = this.rawencode (this.keysort (query));\n                url += '?' + queryString;\n                if (method !== 'GET') {\n                    body = queryString;\n                }\n            }\n            let auth = endpoint + '/' + nonce + '/' + queryString;\n            let payload = this.stringToBase64 (this.encode (auth));\n            // payload should be \"encoded\" as returned from stringToBase64\n            let signature = this.hmac (payload, this.encode (this.secret), 'sha256');\n            headers = {\n                'KC-API-KEY': this.apiKey,\n                'KC-API-NONCE': nonce,\n                'KC-API-SIGNATURE': signature,\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    throwExceptionOnError (response) {\n        if ('success' in response) {\n            if (!response['success']) {\n                if ('code' in response) {\n                    let message = this.safeString (response, 'msg');\n                    if (response['code'] === 'UNAUTH') {\n                        if (message === 'Invalid nonce')\n                            throw new InvalidNonce (this.id + ' ' + message);\n                        throw new AuthenticationError (this.id + ' ' + this.json (response));\n                    } else if (response['code'] === 'ERROR') {\n                        if (message.indexOf ('precision of amount') >= 0)\n                            throw new InvalidOrder (this.id + ' ' + message);\n                        if (message.indexOf ('Min amount each order') >= 0)\n                            throw new InvalidOrder (this.id + ' ' + message);\n                    }\n                }\n            }\n        }\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (body && (body[0] === '{')) {\n            let response = JSON.parse (body);\n            this.throwExceptionOnError (response);\n        }\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        this.throwExceptionOnError (response);\n        return response;\n    }\n}\n","'use strict';\n\n// ---------------------------------------------------------------------------\n\nconst acx = require ('./acx.js');\nconst { ExchangeError, InsufficientFunds, OrderNotFound } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class kuna extends acx {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'kuna',\n            'name': 'Kuna',\n            'countries': 'UA',\n            'rateLimit': 1000,\n            'version': 'v2',\n            'hasCORS': false,\n            'hasFetchTickers': false,\n            'hasFetchOHLCV': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/31697638-912824fa-b3c1-11e7-8c36-cf9606eb94ac.jpg',\n                'api': 'https://kuna.io',\n                'www': 'https://kuna.io',\n                'doc': 'https://kuna.io/documents/api',\n                'fees': 'https://kuna.io/documents/api',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'tickers/{market}',\n                        'order_book',\n                        'order_book/{market}',\n                        'trades',\n                        'trades/{market}',\n                        'timestamp',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'members/me',\n                        'orders',\n                        'trades/my',\n                    ],\n                    'post': [\n                        'orders',\n                        'order/delete',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/UAH': { 'id': 'btcuah', 'symbol': 'BTC/UAH', 'base': 'BTC', 'quote': 'UAH', 'precision': { 'amount': 6, 'price': 0 }, 'lot': 0.000001, 'limits': { 'amount': { 'min': 0.000001, 'max': undefined }, 'price': { 'min': 1, 'max': undefined }, 'cost': { 'min': 0.000001, 'max': undefined }}},\n                'ETH/UAH': { 'id': 'ethuah', 'symbol': 'ETH/UAH', 'base': 'ETH', 'quote': 'UAH', 'precision': { 'amount': 6, 'price': 0 }, 'lot': 0.000001, 'limits': { 'amount': { 'min': 0.000001, 'max': undefined }, 'price': { 'min': 1, 'max': undefined }, 'cost': { 'min': 0.000001, 'max': undefined }}},\n                'GBG/UAH': { 'id': 'gbguah', 'symbol': 'GBG/UAH', 'base': 'GBG', 'quote': 'UAH', 'precision': { 'amount': 3, 'price': 2 }, 'lot': 0.001, 'limits': { 'amount': { 'min': 0.000001, 'max': undefined }, 'price': { 'min': 0.01, 'max': undefined }, 'cost': { 'min': 0.000001, 'max': undefined }}}, // Golos Gold (GBG != GOLOS)\n                'KUN/BTC': { 'id': 'kunbtc', 'symbol': 'KUN/BTC', 'base': 'KUN', 'quote': 'BTC', 'precision': { 'amount': 6, 'price': 6 }, 'lot': 0.000001, 'limits': { 'amount': { 'min': 0.000001, 'max': undefined }, 'price': { 'min': 0.000001, 'max': undefined }, 'cost': { 'min': 0.000001, 'max': undefined }}},\n                'BCH/BTC': { 'id': 'bchbtc', 'symbol': 'BCH/BTC', 'base': 'BCH', 'quote': 'BTC', 'precision': { 'amount': 6, 'price': 6 }, 'lot': 0.000001, 'limits': { 'amount': { 'min': 0.000001, 'max': undefined }, 'price': { 'min': 0.000001, 'max': undefined }, 'cost': { 'min': 0.000001, 'max': undefined }}},\n                'BCH/UAH': { 'id': 'bchuah', 'symbol': 'BCH/UAH', 'base': 'BCH', 'quote': 'UAH', 'precision': { 'amount': 6, 'price': 0 }, 'lot': 0.000001, 'limits': { 'amount': { 'min': 0.000001, 'max': undefined }, 'price': { 'min': 1, 'max': undefined }, 'cost': { 'min': 0.000001, 'max': undefined }}},\n                'WAVES/UAH': { 'id': 'wavesuah', 'symbol': 'WAVES/UAH', 'base': 'WAVES', 'quote': 'UAH', 'precision': { 'amount': 6, 'price': 0 }, 'lot': 0.000001, 'limits': { 'amount': { 'min': 0.000001, 'max': undefined }, 'price': { 'min': 1, 'max': undefined }, 'cost': { 'min': 0.000001, 'max': undefined }}},\n                'ARN/BTC': { 'id': 'arnbtc', 'symbol': 'ARN/BTC', 'base': 'ARN', 'quote': 'BTC' },\n                'B2B/BTC': { 'id': 'b2bbtc', 'symbol': 'B2B/BTC', 'base': 'B2B', 'quote': 'BTC' },\n                'EVR/BTC': { 'id': 'evrbtc', 'symbol': 'EVR/BTC', 'base': 'EVR', 'quote': 'BTC' },\n                'GOL/GBG': { 'id': 'golgbg', 'symbol': 'GOL/GBG', 'base': 'GOL', 'quote': 'GBG' },\n                'R/BTC': { 'id': 'rbtc', 'symbol': 'R/BTC', 'base': 'R', 'quote': 'BTC' },\n                'RMC/BTC': { 'id': 'rmcbtc', 'symbol': 'RMC/BTC', 'base': 'RMC', 'quote': 'BTC' },\n            },\n            'fees': {\n                'trading': {\n                    'taker': 0.25 / 100,\n                    'maker': 0.25 / 100,\n                },\n                'funding': {\n                    'withdraw': {\n                        'UAH': '1%',\n                        'BTC': 0.001,\n                        'BCH': 0.001,\n                        'ETH': 0.01,\n                        'WAVES': 0.01,\n                        'GOL': 0.0,\n                        'GBG': 0.0,\n                        // 'RMC': 0.001 BTC\n                        // 'ARN': 0.01 ETH\n                        // 'R': 0.01 ETH\n                        // 'EVR': 0.01 ETH\n                    },\n                    'deposit': {\n                        // 'UAH': (amount) => amount * 0.001 + 5\n                    },\n                },\n            },\n        });\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code === 400) {\n            let response = JSON.parse (body);\n            let error = this.safeValue (response, 'error');\n            let errorCode = this.safeInteger (error, 'code');\n            if (errorCode === 2002) {\n                throw new InsufficientFunds ([ this.id, method, url, code, reason, body ].join (' '));\n            } else if (errorCode === 2003) {\n                throw new OrderNotFound ([ this.id, method, url, code, reason, body ].join (' '));\n            }\n        }\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let market = this.market (symbol);\n        let orderBook = await this.publicGetOrderBook (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseOrderBook (orderBook, undefined, 'bids', 'asks', 'price', 'remaining_volume');\n    }\n\n    async fetchL3OrderBook (symbol, params) {\n        return this.fetchOrderBook (symbol, params);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOpenOrders requires a symbol argument');\n        let market = this.market (symbol);\n        let orders = await this.privateGetOrders (this.extend ({\n            'market': market['id'],\n        }, params));\n        // todo emulation of fetchClosedOrders, fetchOrders, fetchOrder\n        // with order cache + fetchOpenOrders\n        // as in BTC-e, Liqui, Yobit, DSX, Tidex, WEX\n        return this.parseOrders (orders, market, since, limit);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = this.parse8601 (trade['created_at']);\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'id': trade['id'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': undefined,\n            'side': undefined,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['volume']),\n            'info': trade,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetTrades (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    parseMyTrade (trade, market) {\n        let timestamp = this.parse8601 (trade['created_at']);\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'id': trade['id'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'price': trade['price'],\n            'amount': trade['volume'],\n            'cost': trade['funds'],\n            'symbol': symbol,\n            'side': trade['side'],\n            'order': trade['order_id'],\n        };\n    }\n\n    parseMyTrades (trades, market = undefined) {\n        let parsedTrades = [];\n        for (let i = 0; i < trades.length; i++) {\n            let trade = trades[i];\n            let parsedTrade = this.parseMyTrade (trade, market);\n            parsedTrades.push (parsedTrade);\n        }\n        return parsedTrades;\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOpenOrders requires a symbol argument');\n        let market = this.market (symbol);\n        let response = await this.privateGetTradesMy ({ 'market': market['id'] });\n        return this.parseMyTrades (response, market);\n    }\n};\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class lakebtc extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'lakebtc',\n            'name': 'LakeBTC',\n            'countries': 'US',\n            'version': 'api_v2',\n            'hasCORS': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28074120-72b7c38a-6660-11e7-92d9-d9027502281d.jpg',\n                'api': 'https://api.lakebtc.com',\n                'www': 'https://www.lakebtc.com',\n                'doc': [\n                    'https://www.lakebtc.com/s/api_v2',\n                    'https://www.lakebtc.com/s/api',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'bcorderbook',\n                        'bctrades',\n                        'ticker',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'buyOrder',\n                        'cancelOrders',\n                        'getAccountInfo',\n                        'getExternalAccounts',\n                        'getOrders',\n                        'getTrades',\n                        'openOrders',\n                        'sellOrder',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.15 / 100,\n                    'taker': 0.2 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetTicker ();\n        let result = [];\n        let keys = Object.keys (markets);\n        for (let k = 0; k < keys.length; k++) {\n            let id = keys[k];\n            let market = markets[id];\n            let base = id.slice (0, 3);\n            let quote = id.slice (3, 6);\n            base = base.toUpperCase ();\n            quote = quote.toUpperCase ();\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetAccountInfo ();\n        let balances = response['balance'];\n        let result = { 'info': response };\n        let currencies = Object.keys (balances);\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let balance = parseFloat (balances[currency]);\n            let account = {\n                'free': balance,\n                'used': 0.0,\n                'total': balance,\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetBcorderbook (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let tickers = await this.publicGetTicker (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        let ticker = tickers[market['id']];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'bid'),\n            'ask': this.safeFloat (ticker, 'ask'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'volume'),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['date'] * 1000;\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': trade['tid'].toString (),\n            'order': undefined,\n            'type': undefined,\n            'side': undefined,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetBctrades (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (market, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        if (type == 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let method = 'privatePost' + this.capitalize (side) + 'Order';\n        let marketId = this.marketId (market);\n        let order = {\n            'params': [ price, amount, marketId ],\n        };\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelOrder ({ 'params': id });\n    }\n\n    nonce () {\n        return this.microseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version;\n        if (api == 'public') {\n            url += '/' + path;\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            if (Object.keys (params).length)\n                params = params.join (',');\n            else\n                params = '';\n            let query = this.urlencode ({\n                'tonce': nonce,\n                'accesskey': this.apiKey,\n                'requestmethod': method.toLowerCase (),\n                'id': nonce,\n                'method': path,\n                'params': params,\n            });\n            body = this.json ({\n                'method': path,\n                'params': params,\n                'id': nonce,\n            });\n            let signature = this.hmac (this.encode (query), this.encode (this.secret), 'sha1');\n            let auth = this.encode (this.apiKey + ':' + signature);\n            headers = {\n                'Json-Rpc-Tonce': nonce,\n                'Authorization': \"Basic \" + this.decode (this.stringToBase64 (auth)),\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","'use strict';\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InsufficientFunds, OrderNotFound, DDoSProtection, InvalidOrder, AuthenticationError } = require ('./base/errors');\n\nmodule.exports = class liqui extends Exchange {\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'liqui',\n            'name': 'Liqui',\n            'countries': 'UA',\n            'rateLimit': 3000,\n            'version': '3',\n            'hasCORS': false,\n            'userAgent': this.userAgents['chrome'],\n            // obsolete metainfo interface\n            'hasFetchOrder': true,\n            'hasFetchOrders': true,\n            'hasFetchOpenOrders': true,\n            'hasFetchClosedOrders': true,\n            'hasFetchTickers': true,\n            'hasFetchMyTrades': true,\n            'hasWithdraw': true,\n            // new metainfo interface\n            'has': {\n                'fetchOrder': true,\n                'fetchOrders': 'emulated',\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': 'emulated',\n                'fetchTickers': true,\n                'fetchMyTrades': true,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27982022-75aea828-63a0-11e7-9511-ca584a8edd74.jpg',\n                'api': {\n                    'public': 'https://api.liqui.io/api',\n                    'private': 'https://api.liqui.io/tapi',\n                },\n                'www': 'https://liqui.io',\n                'doc': 'https://liqui.io/api',\n                'fees': 'https://liqui.io/fee',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'info',\n                        'ticker/{pair}',\n                        'depth/{pair}',\n                        'trades/{pair}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'getInfo',\n                        'Trade',\n                        'ActiveOrders',\n                        'OrderInfo',\n                        'CancelOrder',\n                        'TradeHistory',\n                        'CoinDepositAddress',\n                        'WithdrawCoin',\n                        'CreateCoupon',\n                        'RedeemCoupon',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.001,\n                    'taker': 0.0025,\n                },\n                'funding': 0.0,\n            },\n            'exceptions': {\n                '803': InvalidOrder, // \"Count could not be less than 0.001.\" (selling below minAmount)\n                '804': InvalidOrder, // \"Count could not be more than 10000.\" (buying above maxAmount)\n                '805': InvalidOrder, // \"price could not be less than X.\" (minPrice violation on buy & sell)\n                '806': InvalidOrder, // \"price could not be more than X.\" (maxPrice violation on buy & sell)\n                '807': InvalidOrder, // \"cost could not be less than X.\" (minCost violation on buy & sell)\n                '831': InsufficientFunds, // \"Not enougth X to create buy order.\" (buying with balance.quote < order.cost)\n                '832': InsufficientFunds, // \"Not enougth X to create sell order.\" (selling with balance.base < order.amount)\n                '833': OrderNotFound, // \"Order with id X was not found.\" (cancelling non-existent, closed and cancelled order)\n            },\n        });\n    }\n\n    calculateFee (symbol, type, side, amount, price, takerOrMaker = 'taker', params = {}) {\n        let market = this.markets[symbol];\n        let key = 'quote';\n        let rate = market[takerOrMaker];\n        let cost = parseFloat (this.costToPrecision (symbol, amount * rate));\n        if (side === 'sell') {\n            cost *= price;\n        } else {\n            key = 'base';\n        }\n        return {\n            'type': takerOrMaker,\n            'currency': market[key],\n            'rate': rate,\n            'cost': cost,\n        };\n    }\n\n    commonCurrencyCode (currency) {\n        if (!this.substituteCommonCurrencyCodes)\n            return currency;\n        if (currency === 'XBT')\n            return 'BTC';\n        if (currency === 'BCC')\n            return 'BCH';\n        if (currency === 'DRK')\n            return 'DASH';\n        // they misspell DASH as dsh :/\n        if (currency === 'DSH')\n            return 'DASH';\n        return currency;\n    }\n\n    getBaseQuoteFromMarketId (id) {\n        let uppercase = id.toUpperCase ();\n        let [ base, quote ] = uppercase.split ('_');\n        base = this.commonCurrencyCode (base);\n        quote = this.commonCurrencyCode (quote);\n        return [ base, quote ];\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicGetInfo ();\n        let markets = response['pairs'];\n        let keys = Object.keys (markets);\n        let result = [];\n        for (let p = 0; p < keys.length; p++) {\n            let id = keys[p];\n            let market = markets[id];\n            let [ base, quote ] = this.getBaseQuoteFromMarketId (id);\n            let symbol = base + '/' + quote;\n            let precision = {\n                'amount': this.safeInteger (market, 'decimal_places'),\n                'price': this.safeInteger (market, 'decimal_places'),\n            };\n            let amountLimits = {\n                'min': this.safeFloat (market, 'min_amount'),\n                'max': this.safeFloat (market, 'max_amount'),\n            };\n            let priceLimits = {\n                'min': this.safeFloat (market, 'min_price'),\n                'max': this.safeFloat (market, 'max_price'),\n            };\n            let costLimits = {\n                'min': this.safeFloat (market, 'min_total'),\n            };\n            let limits = {\n                'amount': amountLimits,\n                'price': priceLimits,\n                'cost': costLimits,\n            };\n            let hidden = this.safeInteger (market, 'hidden');\n            let active = (hidden === 0);\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': active,\n                'taker': market['fee'] / 100,\n                'lot': amountLimits['min'],\n                'precision': precision,\n                'limits': limits,\n                'info': market,\n            }));\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetInfo ();\n        let balances = response['return'];\n        let result = { 'info': balances };\n        let funds = balances['funds'];\n        let currencies = Object.keys (funds);\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let uppercase = currency.toUpperCase ();\n            uppercase = this.commonCurrencyCode (uppercase);\n            let total = undefined;\n            let used = undefined;\n            if (balances['open_orders'] === 0) {\n                total = funds[currency];\n                used = 0.0;\n            }\n            let account = {\n                'free': funds[currency],\n                'used': used,\n                'total': total,\n            };\n            result[uppercase] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetDepthPair (this.extend ({\n            'pair': market['id'],\n            // 'limit': 150, // default = 150, max = 2000\n        }, params));\n        let market_id_in_reponse = (market['id'] in response);\n        if (!market_id_in_reponse)\n            throw new ExchangeError (this.id + ' ' + market['symbol'] + ' order book is empty or not available');\n        let orderbook = response[market['id']];\n        let result = this.parseOrderBook (orderbook);\n        result['bids'] = this.sortBy (result['bids'], 0, true);\n        result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['updated'] * 1000;\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'buy'),\n            'ask': this.safeFloat (ticker, 'sell'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': this.safeFloat (ticker, 'avg'),\n            'baseVolume': this.safeFloat (ticker, 'vol_cur'),\n            'quoteVolume': this.safeFloat (ticker, 'vol'),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let ids = undefined;\n        if (!symbols) {\n            // let numIds = this.ids.length;\n            // if (numIds > 256)\n            //     throw new ExchangeError (this.id + ' fetchTickers() requires symbols argument');\n            ids = this.ids.join ('-');\n            if (ids.length > 2083) {\n                let numIds = this.ids.length;\n                throw new ExchangeError (this.id + ' has ' + numIds.toString () + ' symbols exceeding max URL length, you are required to specify a list of symbols in the first argument to fetchTickers');\n            }\n        } else {\n            ids = this.marketIds (symbols);\n            ids = ids.join ('-');\n        }\n        let tickers = await this.publicGetTickerPair (this.extend ({\n            'pair': ids,\n        }, params));\n        let result = {};\n        let keys = Object.keys (tickers);\n        for (let k = 0; k < keys.length; k++) {\n            let id = keys[k];\n            let ticker = tickers[id];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let tickers = await this.fetchTickers ([ symbol ], params);\n        return tickers[symbol];\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = parseInt (trade['timestamp']) * 1000;\n        let side = trade['type'];\n        if (side === 'ask')\n            side = 'sell';\n        if (side === 'bid')\n            side = 'buy';\n        let price = this.safeFloat (trade, 'price');\n        if ('rate' in trade)\n            price = this.safeFloat (trade, 'rate');\n        let id = this.safeString (trade, 'tid');\n        if ('trade_id' in trade)\n            id = this.safeString (trade, 'trade_id');\n        let order = this.safeString (trade, this.getOrderIdKey ());\n        if ('pair' in trade) {\n            let marketId = trade['pair'];\n            market = this.markets_by_id[marketId];\n        }\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let amount = trade['amount'];\n        let type = 'limit'; // all trades are still limit trades\n        let fee = undefined;\n        // this is filled by fetchMyTrades() only\n        // is_your_order is always false :\\\n        // let isYourOrder = this.safeValue (trade, 'is_your_order');\n        // let takerOrMaker = 'taker';\n        // if (isYourOrder)\n        //     takerOrMaker = 'maker';\n        // let fee = this.calculateFee (symbol, type, side, amount, price, takerOrMaker);\n        return {\n            'id': id,\n            'order': order,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': price,\n            'amount': amount,\n            'fee': fee,\n            'info': trade,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'pair': market['id'],\n        };\n        if (limit)\n            request['limit'] = limit;\n        let response = await this.publicGetTradesPair (this.extend (request, params));\n        return this.parseTrades (response[market['id']], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type === 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'pair': market['id'],\n            'type': side,\n            'amount': this.amountToPrecision (symbol, amount),\n            'rate': this.priceToPrecision (symbol, price),\n        };\n        let response = await this.privatePostTrade (this.extend (request, params));\n        let id = this.safeString (response['return'], this.getOrderIdKey ());\n        let timestamp = this.milliseconds ();\n        price = parseFloat (price);\n        amount = parseFloat (amount);\n        let status = 'open';\n        if (id === '0') {\n            id = this.safeString (response['return'], 'init_order_id');\n            status = 'closed';\n        }\n        let filled = this.safeFloat (response['return'], 'received', 0.0);\n        let remaining = this.safeFloat (response['return'], 'remains', amount);\n        let order = {\n            'id': id,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': status,\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': price,\n            'cost': price * filled,\n            'amount': amount,\n            'remaining': remaining,\n            'filled': filled,\n            'fee': undefined,\n            // 'trades': this.parseTrades (order['trades'], market),\n        };\n        this.orders[id] = order;\n        return this.extend ({ 'info': response }, order);\n    }\n\n    getOrderIdKey () {\n        return 'order_id';\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = undefined;\n        let request = {};\n        let idKey = this.getOrderIdKey ();\n        request[idKey] = id;\n        response = await this.privatePostCancelOrder (this.extend (request, params));\n        if (id in this.orders)\n            this.orders[id]['status'] = 'canceled';\n        return response;\n    }\n\n    parseOrder (order, market = undefined) {\n        let id = order['id'].toString ();\n        let status = this.safeInteger (order, 'status');\n        if (status === 0) {\n            status = 'open';\n        } else if (status === 1) {\n            status = 'closed';\n        } else if ((status === 2) || (status === 3)) {\n            status = 'canceled';\n        }\n        let timestamp = parseInt (order['timestamp_created']) * 1000;\n        let symbol = undefined;\n        if (!market)\n            market = this.markets_by_id[order['pair']];\n        if (market)\n            symbol = market['symbol'];\n        let remaining = undefined;\n        let amount = undefined;\n        let price = this.safeFloat (order, 'rate');\n        let filled = undefined;\n        let cost = undefined;\n        if ('start_amount' in order) {\n            amount = this.safeFloat (order, 'start_amount');\n            remaining = this.safeFloat (order, 'amount');\n        } else {\n            remaining = this.safeFloat (order, 'amount');\n            if (id in this.orders)\n                amount = this.orders[id]['amount'];\n        }\n        if (typeof amount !== 'undefined') {\n            if (typeof remaining !== 'undefined') {\n                filled = amount - remaining;\n                cost = price * filled;\n            }\n        }\n        let fee = undefined;\n        let result = {\n            'info': order,\n            'id': id,\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'type': 'limit',\n            'side': order['type'],\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'remaining': remaining,\n            'filled': filled,\n            'status': status,\n            'fee': fee,\n        };\n        return result;\n    }\n\n    parseOrders (orders, market = undefined, since = undefined, limit = undefined) {\n        let ids = Object.keys (orders);\n        let result = [];\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let order = orders[id];\n            let extended = this.extend (order, { 'id': id });\n            result.push (this.parseOrder (extended, market));\n        }\n        return this.filterBySinceLimit (result, since, limit);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostOrderInfo (this.extend ({\n            'order_id': parseInt (id),\n        }, params));\n        id = id.toString ();\n        let newOrder = this.parseOrder (this.extend ({ 'id': id }, response['return'][id]));\n        let oldOrder = (id in this.orders) ? this.orders[id] : {};\n        this.orders[id] = this.extend (oldOrder, newOrder);\n        return this.orders[id];\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        // if (!symbol)\n        //     throw new ExchangeError (this.id + ' fetchOrders requires a symbol');\n        await this.loadMarkets ();\n        let request = {};\n        let market = undefined;\n        if (symbol) {\n            let market = this.market (symbol);\n            request['pair'] = market['id'];\n        }\n        let response = await this.privatePostActiveOrders (this.extend (request, params));\n        let openOrders = [];\n        if ('return' in response)\n            openOrders = this.parseOrders (response['return'], market);\n        for (let j = 0; j < openOrders.length; j++) {\n            this.orders[openOrders[j]['id']] = openOrders[j];\n        }\n        let openOrdersIndexedById = this.indexBy (openOrders, 'id');\n        let cachedOrderIds = Object.keys (this.orders);\n        let result = [];\n        for (let k = 0; k < cachedOrderIds.length; k++) {\n            let id = cachedOrderIds[k];\n            if (id in openOrdersIndexedById) {\n                this.orders[id] = this.extend (this.orders[id], openOrdersIndexedById[id]);\n            } else {\n                let order = this.orders[id];\n                if (order['status'] === 'open') {\n                    this.orders[id] = this.extend (order, {\n                        'status': 'closed',\n                        'cost': order['amount'] * order['price'],\n                        'filled': order['amount'],\n                        'remaining': 0.0,\n                    });\n                }\n            }\n            let order = this.orders[id];\n            if (symbol) {\n                if (order['symbol'] === symbol)\n                    result.push (order);\n            } else {\n                result.push (order);\n            }\n        }\n        return this.filterBySinceLimit (result, since, limit);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, since, limit, params);\n        let result = [];\n        for (let i = 0; i < orders.length; i++) {\n            if (orders[i]['status'] === 'open')\n                result.push (orders[i]);\n        }\n        return result;\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, since, limit, params);\n        let result = [];\n        for (let i = 0; i < orders.length; i++) {\n            if (orders[i]['status'] === 'closed')\n                result.push (orders[i]);\n        }\n        return result;\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        let request = {\n            // 'from': 123456789, // trade ID, from which the display starts numerical 0\n            // 'count': 1000, // the number of trades for display numerical, default = 1000\n            // 'from_id': trade ID, from which the display starts numerical 0\n            // 'end_id': trade ID on which the display ends numerical ∞\n            // 'order': 'ASC', // sorting, default = DESC\n            // 'since': 1234567890, // UTC start time, default = 0\n            // 'end': 1234567890, // UTC end time, default = ∞\n            // 'pair': 'eth_btc', // default = all markets\n        };\n        if (symbol) {\n            market = this.market (symbol);\n            request['pair'] = market['id'];\n        }\n        if (limit)\n            request['count'] = parseInt (limit);\n        if (since)\n            request['since'] = parseInt (since / 1000);\n        let response = await this.privatePostTradeHistory (this.extend (request, params));\n        let trades = [];\n        if ('return' in response)\n            trades = response['return'];\n        return this.parseTrades (trades, market, since, limit);\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostWithdrawCoin (this.extend ({\n            'coinName': currency,\n            'amount': parseFloat (amount),\n            'address': address,\n        }, params));\n        return {\n            'info': response,\n            'id': response['return']['tId'],\n        };\n    }\n\n    signBodyWithSecret (body) {\n        return this.hmac (this.encode (body), this.encode (this.secret), 'sha512');\n    }\n\n    getVersionString () {\n        return '/' + this.version;\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({\n                'nonce': nonce,\n                'method': path,\n            }, query));\n            let signature = this.signBodyWithSecret (body);\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Key': this.apiKey,\n                'Sign': signature,\n            };\n        } else {\n            url += this.getVersionString () + '/' + this.implodeParams (path, params);\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (httpCode, reason, url, method, headers, body) {\n        if ((typeof body !== 'string') || (body.length < 2))\n            return; // fallback to default error handler\n        if ((body[0] === '{') || (body[0] === '[')) {\n            let response = JSON.parse (body);\n            if ('success' in response) {\n                //\n                // 1 - Liqui only returns the integer 'success' key from their private API\n                //\n                //     { \"success\": 1, ... } httpCode === 200\n                //     { \"success\": 0, ... } httpCode === 200\n                //\n                // 2 - However, exchanges derived from Liqui, can return non-integers\n                //\n                //     It can be a numeric string\n                //     { \"sucesss\": \"1\", ... }\n                //     { \"sucesss\": \"0\", ... }, httpCode >= 200 (can be 403, 502, etc)\n                //\n                //     Or just a string\n                //     { \"success\": \"true\", ... }\n                //     { \"success\": \"false\", ... }, httpCode >= 200\n                //\n                //     Or a boolean\n                //     { \"success\": true, ... }\n                //     { \"success\": false, ... }, httpCode >= 200\n                //\n                // 3 - Oversimplified, Python PEP8 forbids comparison operator (===) of different types\n                //\n                // 4 - We do not want to copy-paste and duplicate the code of this handler to other exchanges derived from Liqui\n                //\n                // To cover points 1, 2, 3 and 4 combined this handler should work like this:\n                //\n                let success = this.safeValue (response, 'success', false);\n                if (typeof success === 'string') {\n                    if ((success === 'true') || (success === '1'))\n                        success = true;\n                    else\n                        success = false;\n                }\n                if (!success) {\n                    const code = response['code'];\n                    const message = response['error'];\n                    const feedback = this.id + ' ' + this.json (response);\n                    const exceptions = this.exceptions;\n                    if (code in exceptions) {\n                        throw new exceptions[code] (feedback);\n                    }\n                    // need a second error map for these messages, apparently...\n                    // in fact, we can use the same .exceptions with string-keys to save some loc here\n                    if (message === 'invalid api key') {\n                        throw new AuthenticationError (feedback);\n                    } else if (message === 'api key dont have trade permission') {\n                        throw new AuthenticationError (feedback);\n                    } else if (message.indexOf ('invalid parameter') >= 0) { // errorCode 0, returned on buy(symbol, 0, 0)\n                        throw new InvalidOrder (feedback);\n                    } else if (message === 'Requests too often') {\n                        throw new DDoSProtection (feedback);\n                    } else if (message === 'not available') {\n                        throw new DDoSProtection (feedback);\n                    } else if (message === 'external service unavailable') {\n                        throw new DDoSProtection (feedback);\n                    } else {\n                        throw new ExchangeError (this.id + ' unknown \"error\" value: ' + this.json (response));\n                    }\n                }\n            }\n        }\n    }\n};\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, AuthenticationError, NotSupported, InvalidOrder, OrderNotFound, ExchangeNotAvailable } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class livecoin extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'livecoin',\n            'name': 'LiveCoin',\n            'countries': [ 'US', 'UK', 'RU' ],\n            'rateLimit': 1000,\n            'hasCORS': false,\n            // obsolete metainfo interface\n            'hasFetchTickers': true,\n            'hasFetchCurrencies': true,\n            // new metainfo interface\n            'has': {\n                'fetchTickers': true,\n                'fetchCurrencies': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27980768-f22fc424-638a-11e7-89c9-6010a54ff9be.jpg',\n                'api': 'https://api.livecoin.net',\n                'www': 'https://www.livecoin.net',\n                'doc': 'https://www.livecoin.net/api?lang=en',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'exchange/all/order_book',\n                        'exchange/last_trades',\n                        'exchange/maxbid_minask',\n                        'exchange/order_book',\n                        'exchange/restrictions',\n                        'exchange/ticker', // omit params to get all tickers at once\n                        'info/coinInfo',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'exchange/client_orders',\n                        'exchange/order',\n                        'exchange/trades',\n                        'exchange/commission',\n                        'exchange/commissionCommonInfo',\n                        'payment/balances',\n                        'payment/balance',\n                        'payment/get/address',\n                        'payment/history/size',\n                        'payment/history/transactions',\n                    ],\n                    'post': [\n                        'exchange/buylimit',\n                        'exchange/buymarket',\n                        'exchange/cancellimit',\n                        'exchange/selllimit',\n                        'exchange/sellmarket',\n                        'payment/out/capitalist',\n                        'payment/out/card',\n                        'payment/out/coin',\n                        'payment/out/okpay',\n                        'payment/out/payeer',\n                        'payment/out/perfectmoney',\n                        'payment/voucher/amount',\n                        'payment/voucher/make',\n                        'payment/voucher/redeem',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'maker': 0.18 / 100,\n                    'taker': 0.18 / 100,\n                },\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        return currency;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetExchangeTicker ();\n        let restrictions = await this.publicGetExchangeRestrictions ();\n        let restrictionsById = this.indexBy (restrictions['restrictions'], 'currencyPair');\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['symbol'];\n            let symbol = id;\n            let [ base, quote ] = symbol.split ('/');\n            let coinRestrictions = this.safeValue (restrictionsById, symbol);\n            let precision = {\n                'price': 5,\n                'amount': 8,\n                'cost': 8,\n            };\n            let limits = {\n                'amount': {\n                    'min': Math.pow (10, -precision['amount']),\n                    'max': Math.pow (10, precision['amount']),\n                },\n            };\n            if (coinRestrictions) {\n                precision['price'] = this.safeInteger (coinRestrictions, 'priceScale', 5);\n                limits['amount']['min'] = this.safeFloat (coinRestrictions, 'minLimitQuantity', limits['amount']['min']);\n            }\n            limits['price'] = {\n                'min': Math.pow (10, -precision['price']),\n                'max': Math.pow (10, precision['price']),\n            };\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'precision': precision,\n                'limits': limits,\n                'info': market,\n            }));\n        }\n        return result;\n    }\n\n    async fetchCurrencies (params = {}) {\n        let response = await this.publicGetInfoCoinInfo (params);\n        let currencies = response['info'];\n        let result = {};\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let id = currency['symbol'];\n            // todo: will need to rethink the fees\n            // to add support for multiple withdrawal/deposit methods and\n            // differentiated fees for each particular method\n            let code = this.commonCurrencyCode (id);\n            let precision = 8; // default precision, todo: fix \"magic constants\"\n            let active = (currency['walletStatus'] == 'normal');\n            result[code] = {\n                'id': id,\n                'code': code,\n                'info': currency,\n                'name': currency['name'],\n                'active': active,\n                'status': 'ok',\n                'fee': currency['withdrawFee'], // todo: redesign\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': currency['minOrderAmount'],\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': currency['minOrderAmount'],\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': currency['minWithdrawAmount'],\n                        'max': Math.pow (10, precision),\n                    },\n                    'deposit': {\n                        'min': currency['minDepositAmount'],\n                        'max': undefined,\n                    },\n                },\n            };\n        }\n        result = this.appendFiatCurrencies (result);\n        return result;\n    }\n\n    appendFiatCurrencies (result = []) {\n        let precision = 8;\n        let defaults = {\n            'info': undefined,\n            'active': true,\n            'status': 'ok',\n            'fee': undefined,\n            'precision': precision,\n            'limits': {\n                'withdraw': { 'min': undefined, 'max': undefined },\n                'deposit': { 'min': undefined, 'max': undefined },\n                'amount': { 'min': undefined, 'max': undefined },\n                'cost': { 'min': undefined, 'max': undefined },\n                'price': {\n                    'min': Math.pow (10, -precision),\n                    'max': Math.pow (10, precision),\n                },\n            },\n        };\n        let currencies = [\n            { 'id': 'USD', 'code': 'USD', 'name': 'US Dollar' },\n            { 'id': 'EUR', 'code': 'EUR', 'name': 'Euro' },\n            { 'id': 'RUR', 'code': 'RUR', 'name': 'Russian ruble' },\n        ];\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let code = currency['code'];\n            result[code] = this.extend (defaults, currency);\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privateGetPaymentBalances ();\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let account = undefined;\n            if (currency in result)\n                account = result[currency];\n            else\n                account = this.account ();\n            if (balance['type'] == 'total')\n                account['total'] = parseFloat (balance['value']);\n            if (balance['type'] == 'available')\n                account['free'] = parseFloat (balance['value']);\n            if (balance['type'] == 'trade')\n                account['used'] = parseFloat (balance['value']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchFees (params = {}) {\n        await this.loadMarkets ();\n        let commissionInfo = await this.privateGetExchangeCommissionCommonInfo ();\n        let commission = this.safeFloat (commissionInfo, 'commission');\n        return {\n            'info': commissionInfo,\n            'maker': commission,\n            'taker': commission,\n            'withdraw': 0.0,\n        };\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetExchangeOrderBook (this.extend ({\n            'currencyPair': this.marketId (symbol),\n            'groupByPrice': 'false',\n            'depth': 100,\n        }, params));\n        let timestamp = orderbook['timestamp'];\n        return this.parseOrderBook (orderbook, timestamp);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let vwap = parseFloat (ticker['vwap']);\n        let baseVolume = parseFloat (ticker['volume']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['best_bid']),\n            'ask': parseFloat (ticker['best_ask']),\n            'vwap': parseFloat (ticker['vwap']),\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetExchangeTicker (params);\n        let tickers = this.indexBy (response, 'symbol');\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetExchangeTicker (this.extend ({\n            'currencyPair': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['time'] * 1000;\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': trade['id'].toString (),\n            'order': undefined,\n            'type': undefined,\n            'side': trade['type'].toLowerCase (),\n            'price': trade['price'],\n            'amount': trade['quantity'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetExchangeLastTrades (this.extend ({\n            'currencyPair': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    parseOrder (order, market = undefined) {\n        let timestamp = this.safeInteger (order, 'lastModificationTime');\n        if (!timestamp)\n            timestamp = this.parse8601 (order['lastModificationTime']);\n        let trades = undefined;\n        if ('trades' in order)\n            // TODO currently not supported by livecoin\n            // trades = this.parseTrades (order['trades'], market, since, limit);\n            trades = undefined;\n        let status = undefined;\n        if (order['orderStatus'] == 'OPEN' || order['orderStatus'] == 'PARTIALLY_FILLED') {\n            status = 'open';\n        } else if (order['orderStatus'] == 'EXECUTED' || order['orderStatus'] == 'PARTIALLY_FILLED_AND_CANCELLED') {\n            status = 'closed';\n        } else {\n            status = 'canceled';\n        }\n        let symbol = order['currencyPair'];\n        let [ base, quote ] = symbol.split ('/');\n        let type = undefined;\n        let side = undefined;\n        if (order['type'].indexOf ('MARKET') >= 0) {\n            type = 'market';\n        } else {\n            type = 'limit';\n        }\n        if (order['type'].indexOf ('SELL') >= 0) {\n            side = 'sell';\n        } else {\n            side = 'buy';\n        }\n        let price = this.safeFloat (order, 'price', 0.0);\n        let cost = this.safeFloat (order, 'commissionByTrade', 0.0);\n        let remaining = this.safeFloat (order, 'remainingQuantity', 0.0);\n        let amount = this.safeFloat (order, 'quantity', remaining);\n        let filled = amount - remaining;\n        return {\n            'info': order,\n            'id': order['id'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': status,\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'trades': trades,\n            'fee': {\n                'cost': cost,\n                'currency': quote,\n            },\n        };\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        if (symbol)\n            market = this.market (symbol);\n        let pair = market ? market['id'] : undefined;\n        let request = {};\n        if (pair)\n            request['currencyPair'] = pair;\n        if (since)\n            request['issuedFrom'] = parseInt (since);\n        if (limit)\n            request['endRow'] = limit - 1;\n        let response = await this.privateGetExchangeClientOrders (this.extend (request, params));\n        let result = [];\n        let rawOrders = [];\n        if (response['data'])\n            rawOrders = response['data'];\n        for (let i = 0; i < rawOrders.length; i++) {\n            let order = rawOrders[i];\n            result.push (this.parseOrder (order, market));\n        }\n        return result;\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let result = await this.fetchOrders (symbol, since, limit, this.extend ({\n            'openClosed': 'OPEN',\n        }, params));\n        return result;\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let result = await this.fetchOrders (symbol, since, limit, this.extend ({\n            'openClosed': 'CLOSED',\n        }, params));\n        return result;\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let method = 'privatePostExchange' + this.capitalize (side) + type;\n        let market = this.market (symbol);\n        let order = {\n            'quantity': this.amountToPrecision (symbol, amount),\n            'currencyPair': market['id'],\n        };\n        if (type == 'limit')\n            order['price'] = this.priceToPrecision (symbol, price);\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['orderId'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' cancelOrder requires a symbol argument');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let currencyPair = market['id'];\n        let response = await this.privatePostExchangeCancellimit (this.extend ({\n            'orderId': id,\n            'currencyPair': currencyPair,\n        }, params));\n        let message = this.safeString (response, 'message', this.json (response));\n        if ('success' in response) {\n            if (!response['success']) {\n                throw new InvalidOrder (message);\n            } else if ('cancelled' in response) {\n                if (response['cancelled']) {\n                    return response;\n                } else {\n                    throw new OrderNotFound (message);\n                }\n            }\n        }\n        throw new ExchangeError (this.id + ' cancelOrder() failed: ' + this.json (response));\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        let request = {\n            'currency': currency,\n        };\n        let response = await this.privateGetPaymentGetAddress (this.extend (request, params));\n        let address = this.safeString (response, 'wallet');\n        return {\n            'currency': currency,\n            'address': address,\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + path;\n        let query = this.urlencode (this.keysort (params));\n        if (method == 'GET') {\n            if (Object.keys (params).length) {\n                url += '?' + query;\n            }\n        }\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            if (method == 'POST')\n                body = query;\n            let signature = this.hmac (this.encode (query), this.encode (this.secret), 'sha256');\n            headers = {\n                'Api-Key': this.apiKey,\n                'Sign': signature.toUpperCase (),\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code >= 300) {\n            if (body[0] == \"{\") {\n                let response = JSON.parse (body);\n                if ('errorCode' in response) {\n                    let error = response['errorCode'];\n                    if (error == 1) {\n                        throw new ExchangeError (this.id + ' ' + this.json (response));\n                    } else if (error == 2) {\n                        if ('errorMessage' in response) {\n                            if (response['errorMessage'] == 'User not found')\n                                throw new AuthenticationError (this.id + ' ' + response['errorMessage']);\n                        } else {\n                            throw new ExchangeError (this.id + ' ' + this.json (response));\n                        }\n                    } else if ((error == 10) || (error == 11) || (error == 12) || (error == 20) || (error == 30) || (error == 101) || (error == 102)) {\n                        throw new AuthenticationError (this.id + ' ' + this.json (response));\n                    } else if (error == 31) {\n                        throw new NotSupported (this.id + ' ' + this.json (response));\n                    } else if (error == 32) {\n                        throw new ExchangeError (this.id + ' ' + this.json (response));\n                    } else if (error == 100) {\n                        throw new ExchangeError (this.id + ': Invalid parameters ' + this.json (response));\n                    } else if (error == 103) {\n                        throw new InvalidOrder (this.id + ': Invalid currency ' + this.json (response));\n                    } else if (error == 104) {\n                        throw new InvalidOrder (this.id + ': Invalid amount ' + this.json (response));\n                    } else if (error == 105) {\n                        throw new InvalidOrder (this.id + ': Unable to block funds ' + this.json (response));\n                    } else if (error == 503) {\n                        throw new ExchangeNotAvailable (this.id + ': Exchange is not available ' + this.json (response));\n                    } else {\n                        throw new ExchangeError (this.id + ' ' + this.json (response));\n                    }\n                }\n            }\n            throw new ExchangeError (this.id + ' ' + body);\n        }\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('success' in response) {\n            if (!response['success']) {\n                throw new ExchangeError (this.id + ' error: ' + this.json (response));\n            }\n        }\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class luno extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'luno',\n            'name': 'luno',\n            'countries': [ 'GB', 'SG', 'ZA' ],\n            'rateLimit': 10000,\n            'version': '1',\n            'hasCORS': false,\n            'hasFetchTickers': true,\n            'hasFetchOrder': true,\n            'has': {\n                'fetchTickers': true,\n                'fetchOrder': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766607-8c1a69d8-5ede-11e7-930c-540b5eb9be24.jpg',\n                'api': 'https://api.mybitx.com/api',\n                'www': 'https://www.luno.com',\n                'doc': [\n                    'https://www.luno.com/en/api',\n                    'https://npmjs.org/package/bitx',\n                    'https://github.com/bausmeier/node-bitx',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'orderbook',\n                        'ticker',\n                        'tickers',\n                        'trades',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'accounts/{id}/pending',\n                        'accounts/{id}/transactions',\n                        'balance',\n                        'fee_info',\n                        'funding_address',\n                        'listorders',\n                        'listtrades',\n                        'orders/{id}',\n                        'quotes/{id}',\n                        'withdrawals',\n                        'withdrawals/{id}',\n                    ],\n                    'post': [\n                        'accounts',\n                        'postorder',\n                        'marketorder',\n                        'stoporder',\n                        'funding_address',\n                        'withdrawals',\n                        'send',\n                        'quotes',\n                        'oauth2/grant',\n                    ],\n                    'put': [\n                        'quotes/{id}',\n                    ],\n                    'delete': [\n                        'quotes/{id}',\n                        'withdrawals/{id}',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetTickers ();\n        let result = [];\n        for (let p = 0; p < markets['tickers'].length; p++) {\n            let market = markets['tickers'][p];\n            let id = market['pair'];\n            let base = id.slice (0, 3);\n            let quote = id.slice (3, 6);\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetBalance ();\n        let balances = response['balance'];\n        let result = { 'info': response };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = this.commonCurrencyCode (balance['asset']);\n            let reserved = parseFloat (balance['reserved']);\n            let unconfirmed = parseFloat (balance['unconfirmed']);\n            let account = {\n                'free': 0.0,\n                'used': this.sum (reserved, unconfirmed),\n                'total': parseFloat (balance['balance']),\n            };\n            account['free'] = account['total'] - account['used'];\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderbook (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        let timestamp = orderbook['timestamp'];\n        return this.parseOrderBook (orderbook, timestamp, 'bids', 'asks', 'price', 'volume');\n    }\n\n    parseOrder (order, market = undefined) {\n        let timestamp = order['creation_timestamp'];\n        let status = (order['state'] == 'PENDING') ? 'open' : 'closed';\n        let side = (order['type'] == 'ASK') ? 'sell' : 'buy';\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let price = this.safeFloat (order, 'limit_price');\n        let amount = this.safeFloat (order, 'limit_volume');\n        let quoteFee = this.safeFloat (order, 'fee_counter');\n        let baseFee = this.safeFloat (order, 'fee_base');\n        let fee = { 'currency': undefined };\n        if (quoteFee) {\n            fee['side'] = 'quote';\n            fee['cost'] = quoteFee;\n        } else {\n            fee['side'] = 'base';\n            fee['cost'] = baseFee;\n        }\n        return {\n            'id': order['order_id'],\n            'datetime': this.iso8601 (timestamp),\n            'timestamp': timestamp,\n            'status': status,\n            'symbol': symbol,\n            'type': undefined,\n            'side': side,\n            'price': price,\n            'amount': amount,\n            'filled': undefined,\n            'remaining': undefined,\n            'trades': undefined,\n            'fee': fee,\n            'info': order,\n        };\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetOrdersId (this.extend ({\n            'id': id,\n        }, params));\n        return this.parseOrder (response);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['timestamp'];\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last_trade']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['rolling_24_hour_volume']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetTickers (params);\n        let tickers = this.indexBy (response['tickers'], 'pair');\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetTicker (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let side = (trade['is_buy']) ? 'buy' : 'sell';\n        return {\n            'info': trade,\n            'id': undefined,\n            'order': undefined,\n            'timestamp': trade['timestamp'],\n            'datetime': this.iso8601 (trade['timestamp']),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': side,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['volume']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTrades (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTrades (response['trades'], market, since, limit);\n    }\n\n    async createOrder (market, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let method = 'privatePost';\n        let order = { 'pair': this.marketId (market) };\n        if (type == 'market') {\n            method += 'Marketorder';\n            order['type'] = side.toUpperCase ();\n            if (side == 'buy')\n                order['counter_volume'] = amount;\n            else\n                order['base_volume'] = amount;\n        } else {\n            method += 'Order';\n            order['volume'] = amount;\n            order['price'] = price;\n            if (side == 'buy')\n                order['type'] = 'BID';\n            else\n                order['type'] = 'ASK';\n        }\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['order_id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostStoporder ({ 'order_id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (Object.keys (query).length)\n            url += '?' + this.urlencode (query);\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            let auth = this.encode (this.apiKey + ':' + this.secret);\n            auth = this.stringToBase64 (auth);\n            headers = { 'Authorization': 'Basic ' + this.decode (auth) };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class lykke extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'lykke',\n            'name': 'Lykke',\n            'countries': 'CH',\n            'version': 'v1',\n            'rateLimit': 200,\n            'hasCORS': false,\n            // obsolete metainfo interface\n            'hasFetchTrades': false,\n            'hasFetchOHLCV': false,\n            // new metainfo interface\n            'has': {\n                'fetchOHLCV': false,\n                'fetchTrades': false,\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/34487620-3139a7b0-efe6-11e7-90f5-e520cef74451.jpg',\n                'api': {\n                    'mobile': 'https://api.lykkex.com/api',\n                    'public': 'https://hft-api.lykke.com/api',\n                    'private': 'https://hft-api.lykke.com/api',\n                    'test': {\n                        'mobile': 'https://api.lykkex.com/api',\n                        'public': 'https://hft-service-dev.lykkex.net/api',\n                        'private': 'https://hft-service-dev.lykkex.net/api',\n                    },\n                },\n                'www': 'https://www.lykke.com',\n                'doc': [\n                    'https://hft-api.lykke.com/swagger/ui/',\n                    'https://www.lykke.com/lykke_api',\n                ],\n                'fees': 'https://www.lykke.com/trading-conditions',\n            },\n            'api': {\n                'mobile': {\n                    'get': [\n                        'AllAssetPairRates/{market}',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'AssetPairs',\n                        'AssetPairs/{id}',\n                        'IsAlive',\n                        'OrderBooks',\n                        'OrderBooks/{AssetPairId}',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'Orders',\n                        'Orders/{id}',\n                        'Wallets',\n                    ],\n                    'post': [\n                        'Orders/limit',\n                        'Orders/market',\n                        'Orders/{id}/Cancel',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'maker': 0.0010,\n                    'taker': 0.0019,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.001,\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privateGetWallets ();\n        let result = { 'info': balances };\n        for (let i = 0; i < balances.length; i++) {\n            let balance = balances[i];\n            let currency = balance['AssetId'];\n            let total = balance['Balance'];\n            let used = balance['Reserved'];\n            let free = total - used;\n            result[currency] = {\n                'free': free,\n                'used': used,\n                'total': total,\n            };\n        }\n        return this.parseBalance (result);\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostOrdersIdCancel ({ 'id': id });\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let query = {\n            'AssetPairId': market['id'],\n            'OrderAction': this.capitalize (side),\n            'Volume': amount,\n        };\n        if (type == 'market') {\n            query['Asset'] = (side == 'buy') ? market['base'] : market['quote'];\n        } else if (type == 'limit') {\n            query['Price'] = price;\n        }\n        let method = 'privatePostOrders' + this.capitalize (type);\n        let result = await this[method] (this.extend (query, params));\n        return {\n            'id': undefined,\n            'info': result,\n        };\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetAssetPairs ();\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let id = market['Id'];\n            let base = market['BaseAssetId'];\n            let quote = market['QuotingAssetId'];\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = market['Name'];\n            let precision = {\n                'amount': market['Accuracy'],\n                'price': market['InvertedAccuracy'],\n            };\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': true,\n                'info': market,\n                'lot': Math.pow (10, -precision['amount']),\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision['amount']),\n                        'max': Math.pow (10, precision['amount']),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision['price']),\n                        'max': Math.pow (10, precision['price']),\n                    },\n                },\n            }));\n        }\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        ticker = ticker['Result'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['Rate']['Bid']),\n            'ask': parseFloat (ticker['Rate']['Ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': undefined,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.mobileGetAllAssetPairRatesMarket (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseOrderStatus (status) {\n        if (status == 'Pending') {\n            return 'open';\n        } else if (status == 'InOrderBook') {\n            return 'open';\n        } else if (status == 'Processing') {\n            return 'open';\n        } else if (status == 'Matched') {\n            return 'closed';\n        } else if (status == 'Cancelled') {\n            return 'canceled';\n        } else if (status == 'NotEnoughFunds') {\n            return 'NotEnoughFunds';\n        } else if (status == 'NoLiquidity') {\n            return 'NoLiquidity';\n        } else if (status == 'UnknownAsset') {\n            return 'UnknownAsset';\n        } else if (status == 'LeadToNegativeSpread') {\n            return 'LeadToNegativeSpread';\n        }\n        return status;\n    }\n\n    parseOrder (order, market = undefined) {\n        let status = this.parseOrderStatus (order['Status']);\n        let symbol = undefined;\n        if (!market) {\n            if ('AssetPairId' in order)\n                if (order['AssetPairId'] in this.markets_by_id)\n                    market = this.markets_by_id[order['AssetPairId']];\n        }\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = undefined;\n        if ('LastMatchTime' in order) {\n            timestamp = this.parse8601 (order['LastMatchTime']);\n        } else if ('Registered' in order) {\n            timestamp = this.parse8601 (order['Registered']);\n        } else if ('CreatedAt' in order) {\n            timestamp = this.parse8601 (order['CreatedAt']);\n        }\n        let price = this.safeFloat (order, 'Price');\n        let amount = this.safeFloat (order, 'Volume');\n        let remaining = this.safeFloat (order, 'RemainingVolume');\n        let filled = amount - remaining;\n        let cost = filled * price;\n        let result = {\n            'info': order,\n            'id': order['Id'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': undefined,\n            'side': undefined,\n            'price': price,\n            'cost': cost,\n            'average': undefined,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': undefined,\n        };\n        return result;\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        let response = await this.privateGetOrdersId (this.extend ({\n            'id': id,\n        }, params));\n        return this.parseOrder (response);\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let response = await this.privateGetOrders ();\n        return this.parseOrders (response, undefined, since, limit);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let response = await this.privateGetOrders (this.extend ({\n            'status': 'InOrderBook',\n        }, params));\n        return this.parseOrders (response, undefined, since, limit);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let response = await this.privateGetOrders (this.extend ({\n            'status': 'Matched',\n        }, params));\n        return this.parseOrders (response, undefined, since, limit);\n    }\n\n    async fetchOrderBook (symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetOrderBooksAssetPairId (this.extend ({\n            'AssetPairId': this.marketId (symbol),\n        }, params));\n        let orderbook = {\n            'timestamp': undefined,\n            'bids': [],\n            'asks': [],\n        };\n        let timestamp = undefined;\n        for (let i = 0; i < response.length; i++) {\n            let side = response[i];\n            if (side['IsBuy']) {\n                orderbook['bids'] = this.arrayConcat (orderbook['bids'], side['Prices']);\n            } else {\n                orderbook['asks'] = this.arrayConcat (orderbook['asks'], side['Prices']);\n            }\n            let timestamp = this.parse8601 (side['Timestamp']);\n            if (!orderbook['timestamp']) {\n                orderbook['timestamp'] = timestamp;\n            } else {\n                orderbook['timestamp'] = Math.max (orderbook['timestamp'], timestamp);\n            }\n        }\n        if (!timestamp)\n            timestamp = this.milliseconds ();\n        return this.parseOrderBook (orderbook, orderbook['timestamp'], 'bids', 'asks', 'Price', 'Volume');\n    }\n\n    parseBidAsk (bidask, priceKey = 0, amountKey = 1) {\n        let price = parseFloat (bidask[priceKey]);\n        let amount = parseFloat (bidask[amountKey]);\n        if (amount < 0)\n            amount = -amount;\n        return [ price, amount ];\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else if (api == 'private') {\n            if (method == 'GET')\n                if (Object.keys (query).length)\n                    url += '?' + this.urlencode (query);\n            this.checkRequiredCredentials ();\n            headers = {\n                'api-key': this.apiKey,\n                'Accept': 'application/json',\n                'Content-Type': 'application/json',\n            };\n            if (method == 'POST')\n                if (Object.keys (params).length)\n                    body = this.json (params);\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class mercado extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'mercado',\n            'name': 'Mercado Bitcoin',\n            'countries': 'BR', // Brazil\n            'rateLimit': 1000,\n            'version': 'v3',\n            'hasCORS': true,\n            'hasWithdraw': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27837060-e7c58714-60ea-11e7-9192-f05e86adb83f.jpg',\n                'api': {\n                    'public': 'https://www.mercadobitcoin.net/api',\n                    'private': 'https://www.mercadobitcoin.net/tapi',\n                },\n                'www': 'https://www.mercadobitcoin.com.br',\n                'doc': [\n                    'https://www.mercadobitcoin.com.br/api-doc',\n                    'https://www.mercadobitcoin.com.br/trade-api',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        '{coin}/orderbook/', // last slash critical\n                        '{coin}/ticker/',\n                        '{coin}/trades/',\n                        '{coin}/trades/{from}/',\n                        '{coin}/trades/{from}/{to}',\n                        '{coin}/day-summary/{year}/{month}/{day}/',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'cancel_order',\n                        'get_account_info',\n                        'get_order',\n                        'get_withdrawal',\n                        'list_system_messages',\n                        'list_orders',\n                        'list_orderbook',\n                        'place_buy_order',\n                        'place_sell_order',\n                        'withdraw_coin',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/BRL': { 'id': 'BRLBTC', 'symbol': 'BTC/BRL', 'base': 'BTC', 'quote': 'BRL', 'suffix': 'Bitcoin' },\n                'LTC/BRL': { 'id': 'BRLLTC', 'symbol': 'LTC/BRL', 'base': 'LTC', 'quote': 'BRL', 'suffix': 'Litecoin' },\n                'BCH/BRL': { 'id': 'BRLBCH', 'symbol': 'BCH/BRL', 'base': 'BCH', 'quote': 'BRL', 'suffix': 'BCash' },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.3 / 100,\n                    'taker': 0.7 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let market = this.market (symbol);\n        let orderbook = await this.publicGetCoinOrderbook (this.extend ({\n            'coin': market['base'],\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetCoinTicker (this.extend ({\n            'coin': market['base'],\n        }, params));\n        let ticker = response['ticker'];\n        let timestamp = parseInt (ticker['date']) * 1000;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy']),\n            'ask': parseFloat (ticker['sell']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['vol']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['date'] * 1000;\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': trade['tid'].toString (),\n            'order': undefined,\n            'type': undefined,\n            'side': trade['type'],\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetCoinTrades (this.extend ({\n            'coin': market['base'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostGetAccountInfo ();\n        let balances = response['response_data']['balance'];\n        let result = { 'info': response };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let account = this.account ();\n            if (lowercase in balances) {\n                account['free'] = parseFloat (balances[lowercase]['available']);\n                account['total'] = parseFloat (balances[lowercase]['total']);\n                account['used'] = account['total'] - account['free'];\n            }\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type === 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let method = 'privatePostPlace' + this.capitalize (side) + 'Order';\n        let order = {\n            'coin_pair': this.marketId (symbol),\n            'quantity': amount,\n            'limit_price': price,\n        };\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['response_data']['order']['order_id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' cancelOrder() requires a symbol argument');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        return await this.privatePostCancelOrder (this.extend ({\n            'coin_pair': market['id'],\n            'order_id': id,\n        }, params));\n    }\n\n    parseOrder (order, market = undefined) {\n        let side = undefined;\n        if ('order_type' in order)\n            side = (order['order_type'] === 1) ? 'buy' : 'sell';\n        let status = order['status'];\n        let symbol = undefined;\n        if (!market) {\n            if ('coin_pair' in order)\n                if (order['coin_pair'] in this.markets_by_id)\n                    market = this.markets_by_id[order['coin_pair']];\n        }\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = undefined;\n        if ('created_timestamp' in order)\n            timestamp = parseInt (order['created_timestamp']) * 1000;\n        if ('updated_timestamp' in order)\n            timestamp = parseInt (order['updated_timestamp']) * 1000;\n        let fee = {\n            'cost': parseFloat (order['fee']),\n            'currency': market['quote'],\n        };\n        let price = this.safeFloat (order, 'limit_price');\n        // price = this.safeFloat (order, 'executed_price_avg', price);\n        let average = this.safeFloat (order, 'executed_price_avg');\n        let amount = this.safeFloat (order, 'quantity');\n        let filled = this.safeFloat (order, 'executed_quantity');\n        let remaining = amount - filled;\n        let cost = amount * average;\n        let result = {\n            'info': order,\n            'id': order['order_id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': 'limit',\n            'side': side,\n            'price': price,\n            'cost': cost,\n            'average': average,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': fee,\n        };\n        return result;\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' cancelOrder() requires a symbol argument');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = undefined;\n        response = await this.privatePostGetOrder (this.extend ({\n            'coin_pair': market['id'],\n            'order_id': parseInt (id),\n        }, params));\n        return this.parseOrder (response['response_data']['order']);\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            'coin': currency,\n            'quantity': amount.toFixed (10),\n            'address': address,\n        };\n        if (currency === 'BRL') {\n            let account_ref = ('account_ref' in params);\n            if (!account_ref)\n                throw new ExchangeError (this.id + ' requires account_ref parameter to withdraw ' + currency);\n        } else if (currency !== 'LTC') {\n            let tx_fee = ('tx_fee' in params);\n            if (!tx_fee)\n                throw new ExchangeError (this.id + ' requires tx_fee parameter to withdraw ' + currency);\n        }\n        let response = await this.privatePostWithdrawCoin (this.extend (request, params));\n        return {\n            'info': response,\n            'id': response['response_data']['withdrawal']['id'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api] + '/';\n        if (api === 'public') {\n            url += this.implodeParams (path, params);\n        } else {\n            this.checkRequiredCredentials ();\n            url += this.version + '/';\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({\n                'tapi_method': path,\n                'tapi_nonce': nonce,\n            }, params));\n            let auth = '/tapi/' + this.version + '/' + '?' + body;\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'TAPI-ID': this.apiKey,\n                'TAPI-MAC': this.hmac (this.encode (auth), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error_message' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class mixcoins extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'mixcoins',\n            'name': 'MixCoins',\n            'countries': [ 'GB', 'HK' ],\n            'rateLimit': 1500,\n            'version': 'v1',\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30237212-ed29303c-9535-11e7-8af8-fcd381cfa20c.jpg',\n                'api': 'https://mixcoins.com/api',\n                'www': 'https://mixcoins.com',\n                'doc': 'https://mixcoins.com/help/api/',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'ticker',\n                        'trades',\n                        'depth',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'cancel',\n                        'info',\n                        'orders',\n                        'order',\n                        'transactions',\n                        'trade',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/USD': { 'id': 'btc_usd', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD', 'maker': 0.0015, 'taker': 0.0025 },\n                'ETH/BTC': { 'id': 'eth_btc', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC', 'maker': 0.001, 'taker': 0.0015 },\n                'BCH/BTC': { 'id': 'bch_btc', 'symbol': 'BCH/BTC', 'base': 'BCH', 'quote': 'BTC', 'maker': 0.001, 'taker': 0.0015 },\n                'LSK/BTC': { 'id': 'lsk_btc', 'symbol': 'LSK/BTC', 'base': 'LSK', 'quote': 'BTC', 'maker': 0.0015, 'taker': 0.0025 },\n                'BCH/USD': { 'id': 'bch_usd', 'symbol': 'BCH/USD', 'base': 'BCH', 'quote': 'USD', 'maker': 0.001, 'taker': 0.0015 },\n                'ETH/USD': { 'id': 'eth_usd', 'symbol': 'ETH/USD', 'base': 'ETH', 'quote': 'USD', 'maker': 0.001, 'taker': 0.0015 },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostInfo ();\n        let balance = response['result']['wallet'];\n        let result = { 'info': balance };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let account = this.account ();\n            if (lowercase in balance) {\n                account['free'] = parseFloat (balance[lowercase]['avail']);\n                account['used'] = parseFloat (balance[lowercase]['lock']);\n                account['total'] = this.sum (account['free'], account['used']);\n            }\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let response = await this.publicGetDepth (this.extend ({\n            'market': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (response['result']);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let response = await this.publicGetTicker (this.extend ({\n            'market': this.marketId (symbol),\n        }, params));\n        let ticker = response['result'];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy']),\n            'ask': parseFloat (ticker['sell']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['vol']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        return {\n            'id': trade['id'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetTrades (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseTrades (response['result'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let order = {\n            'market': this.marketId (symbol),\n            'op': side,\n            'amount': amount,\n        };\n        if (type == 'market') {\n            order['order_type'] = 1;\n            order['price'] = price;\n        } else {\n            order['order_type'] = 0;\n        }\n        let response = await this.privatePostTrade (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['result']['id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancel ({ 'id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + path;\n        if (api == 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({\n                'nonce': nonce,\n            }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Key': this.apiKey,\n                'Sign': this.hmac (this.encode (body), this.secret, 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('status' in response)\n            if (response['status'] == 200)\n                return response;\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class nova extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'nova',\n            'name': 'Novaexchange',\n            'countries': 'TZ', // Tanzania\n            'rateLimit': 2000,\n            'version': 'v2',\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30518571-78ca0bca-9b8a-11e7-8840-64b83a4a94b2.jpg',\n                'api': 'https://novaexchange.com/remote',\n                'www': 'https://novaexchange.com',\n                'doc': 'https://novaexchange.com/remote/faq',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'markets/',\n                        'markets/{basecurrency}/',\n                        'market/info/{pair}/',\n                        'market/orderhistory/{pair}/',\n                        'market/openorders/{pair}/buy/',\n                        'market/openorders/{pair}/sell/',\n                        'market/openorders/{pair}/both/',\n                        'market/openorders/{pair}/{ordertype}/',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'getbalances/',\n                        'getbalance/{currency}/',\n                        'getdeposits/',\n                        'getwithdrawals/',\n                        'getnewdepositaddress/{currency}/',\n                        'getdepositaddress/{currency}/',\n                        'myopenorders/',\n                        'myopenorders_market/{pair}/',\n                        'cancelorder/{orderid}/',\n                        'withdraw/{currency}/',\n                        'trade/{pair}/',\n                        'tradehistory/',\n                        'getdeposithistory/',\n                        'getwithdrawalhistory/',\n                        'walletstatus/',\n                        'walletstatus/{currency}/',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicGetMarkets ();\n        let markets = response['markets'];\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            if (!market['disabled']) {\n                let id = market['marketname'];\n                let [ quote, base ] = id.split ('_');\n                let symbol = base + '/' + quote;\n                result.push ({\n                    'id': id,\n                    'symbol': symbol,\n                    'base': base,\n                    'quote': quote,\n                    'info': market,\n                });\n            }\n        }\n        return result;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetMarketOpenordersPairBoth (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'buyorders', 'sellorders', 'price', 'amount');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetMarketInfoPair (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        let ticker = response['markets'][0];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high24h']),\n            'low': parseFloat (ticker['low24h']),\n            'bid': this.safeFloat (ticker, 'bid'),\n            'ask': this.safeFloat (ticker, 'ask'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last_price']),\n            'change': parseFloat (ticker['change24h']),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': parseFloat (ticker['volume24h']),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['unix_t_datestamp'] * 1000;\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': undefined,\n            'order': undefined,\n            'type': undefined,\n            'side': trade['tradetype'].toLowerCase (),\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetMarketOrderhistoryPair (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTrades (response['items'], market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetbalances ();\n        let balances = response['balances'];\n        let result = { 'info': response };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let lockbox = parseFloat (balance['amount_lockbox']);\n            let trades = parseFloat (balance['amount_trades']);\n            let account = {\n                'free': parseFloat (balance['amount']),\n                'used': this.sum (lockbox, trades),\n                'total': parseFloat (balance['amount_total']),\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type == 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        await this.loadMarkets ();\n        amount = amount.toString ();\n        price = price.toString ();\n        let market = this.market (symbol);\n        let order = {\n            'tradetype': side.toUpperCase (),\n            'tradeamount': amount,\n            'tradeprice': price,\n            'tradebase': 1,\n            'pair': market['id'],\n        };\n        let response = await this.privatePostTradePair (this.extend (order, params));\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelorder (this.extend ({\n            'orderid': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/';\n        if (api == 'private')\n            url += api + '/';\n        url += this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            url += '?' + this.urlencode ({ 'nonce': nonce });\n            let signature = this.hmac (this.encode (url), this.encode (this.secret), 'sha512', 'base64');\n            body = this.urlencode (this.extend ({\n                'apikey': this.apiKey,\n                'signature': signature,\n            }, query));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('status' in response)\n            if (response['status'] != 'success')\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst okcoinusd = require ('./okcoinusd.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class okcoincny extends okcoinusd {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'okcoincny',\n            'name': 'OKCoin CNY',\n            'countries': 'CN',\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766792-8be9157a-5ee5-11e7-926c-6d69b8d3378d.jpg',\n                'api': {\n                    'web': 'https://www.okcoin.cn',\n                    'public': 'https://www.okcoin.cn/pai',\n                    'private': 'https://www.okcoin.cn/api',\n                },\n                'www': 'https://www.okcoin.cn',\n                'doc': 'https://www.okcoin.cn/rest_getStarted.html',\n            },\n            'markets': {\n                'BTC/CNY': { 'id': 'btc_cny', 'symbol': 'BTC/CNY', 'base': 'BTC', 'quote': 'CNY', 'type': 'spot', 'spot': true, 'future': false },\n                'LTC/CNY': { 'id': 'ltc_cny', 'symbol': 'LTC/CNY', 'base': 'LTC', 'quote': 'CNY', 'type': 'spot', 'spot': true, 'future': false },\n                'ETH/CNY': { 'id': 'eth_cny', 'symbol': 'ETH/CNY', 'base': 'ETH', 'quote': 'CNY', 'type': 'spot', 'spot': true, 'future': false },\n                'ETC/CNY': { 'id': 'etc_cny', 'symbol': 'ETC/CNY', 'base': 'ETC', 'quote': 'CNY', 'type': 'spot', 'spot': true, 'future': false },\n                'BCH/CNY': { 'id': 'bcc_cny', 'symbol': 'BCH/CNY', 'base': 'BCH', 'quote': 'CNY', 'type': 'spot', 'spot': true, 'future': false },\n            },\n        });\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InsufficientFunds, InvalidOrder, OrderNotFound, AuthenticationError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class okcoinusd extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'okcoinusd',\n            'name': 'OKCoin USD',\n            'countries': [ 'CN', 'US' ],\n            'hasCORS': false,\n            'version': 'v1',\n            'rateLimit': 1000, // up to 3000 requests per 5 minutes ≈ 600 requests per minute ≈ 10 requests per second ≈ 100 ms\n            // obsolete metainfo interface\n            'hasFetchOHLCV': true,\n            'hasFetchOrder': true,\n            'hasFetchOrders': false,\n            'hasFetchOpenOrders': true,\n            'hasFetchClosedOrders': true,\n            'hasWithdraw': true,\n            // new metainfo interface\n            'has': {\n                'fetchOHLCV': true,\n                'fetchOrder': true,\n                'fetchOrders': false,\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': true,\n                'withdraw': true,\n            },\n            'extension': '.do', // appended to endpoint URL\n            'hasFutureMarkets': false,\n            'timeframes': {\n                '1m': '1min',\n                '3m': '3min',\n                '5m': '5min',\n                '15m': '15min',\n                '30m': '30min',\n                '1h': '1hour',\n                '2h': '2hour',\n                '4h': '4hour',\n                '6h': '6hour',\n                '12h': '12hour',\n                '1d': '1day',\n                '3d': '3day',\n                '1w': '1week',\n            },\n            'api': {\n                'web': {\n                    'get': [\n                        'markets/currencies',\n                        'markets/products',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'depth',\n                        'exchange_rate',\n                        'future_depth',\n                        'future_estimated_price',\n                        'future_hold_amount',\n                        'future_index',\n                        'future_kline',\n                        'future_price_limit',\n                        'future_ticker',\n                        'future_trades',\n                        'kline',\n                        'otcs',\n                        'ticker',\n                        'trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'account_records',\n                        'batch_trade',\n                        'borrow_money',\n                        'borrow_order_info',\n                        'borrows_info',\n                        'cancel_borrow',\n                        'cancel_order',\n                        'cancel_otc_order',\n                        'cancel_withdraw',\n                        'future_batch_trade',\n                        'future_cancel',\n                        'future_devolve',\n                        'future_explosive',\n                        'future_order_info',\n                        'future_orders_info',\n                        'future_position',\n                        'future_position_4fix',\n                        'future_trade',\n                        'future_trades_history',\n                        'future_userinfo',\n                        'future_userinfo_4fix',\n                        'lend_depth',\n                        'order_fee',\n                        'order_history',\n                        'order_info',\n                        'orders_info',\n                        'otc_order_history',\n                        'otc_order_info',\n                        'repayment',\n                        'submit_otc_order',\n                        'trade',\n                        'trade_history',\n                        'trade_otc_order',\n                        'withdraw',\n                        'withdraw_info',\n                        'unrepayments_info',\n                        'userinfo',\n                    ],\n                },\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766791-89ffb502-5ee5-11e7-8a5b-c5950b68ac65.jpg',\n                'api': {\n                    'web': 'https://www.okcoin.com/v2',\n                    'public': 'https://www.okcoin.com/api',\n                    'private': 'https://www.okcoin.com/api',\n                },\n                'www': 'https://www.okcoin.com',\n                'doc': [\n                    'https://www.okcoin.com/rest_getStarted.html',\n                    'https://www.npmjs.com/package/okcoin.com',\n                ],\n            },\n            'fees': {\n                'trading': {\n                    'taker': 0.002,\n                    'maker': 0.002,\n                },\n            },\n            'exceptions': {\n                '1009': OrderNotFound,\n                '1013': InvalidOrder, // no order type\n                '1027': InvalidOrder, // createLimitBuyOrder(symbol, 0, 0): Incorrect parameter may exceeded limits\n                '1002': InsufficientFunds, // The transaction amount exceed the balance\n                '10000': ExchangeError, // createLimitBuyOrder(symbol, undefined, undefined)\n                '10005': AuthenticationError, // bad apiKey\n                '10008': ExchangeError, // Illegal URL parameter\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.webGetMarketsProducts ();\n        let markets = response['data'];\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let id = markets[i]['symbol'];\n            let uppercase = id.toUpperCase ();\n            let [ base, quote ] = uppercase.split ('_');\n            let symbol = base + '/' + quote;\n            let precision = {\n                'amount': markets[i]['maxSizeDigit'],\n                'price': markets[i]['maxPriceDigit'],\n            };\n            let lot = Math.pow (10, -precision['amount']);\n            let minAmount = markets[i]['minTradeSize'];\n            let minPrice = Math.pow (10, -precision['price']);\n            let market = this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': markets[i],\n                'type': 'spot',\n                'spot': true,\n                'future': false,\n                'lot': lot,\n                'active': true,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': minAmount,\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': minPrice,\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': minAmount * minPrice,\n                        'max': undefined,\n                    },\n                },\n            });\n            result.push (market);\n            if ((this.hasFutureMarkets) && (market['quote'] === 'USDT')) {\n                result.push (this.extend (market, {\n                    'quote': 'USD',\n                    'symbol': market['base'] + '/USD',\n                    'id': market['id'].replace ('usdt', 'usd'),\n                    'type': 'future',\n                    'spot': false,\n                    'future': true,\n                }));\n            }\n        }\n        return result;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'publicGet';\n        let request = {\n            'symbol': market['id'],\n        };\n        if (market['future']) {\n            method += 'Future';\n            request['contract_type'] = 'this_week'; // next_week, quarter\n        }\n        method += 'Depth';\n        let orderbook = await this[method] (this.extend (request, params));\n        let timestamp = this.milliseconds ();\n        return {\n            'bids': orderbook['bids'],\n            'asks': this.sortBy (orderbook['asks'], 0),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n        };\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['timestamp'];\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy']),\n            'ask': parseFloat (ticker['sell']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['vol']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'publicGet';\n        let request = {\n            'symbol': market['id'],\n        };\n        if (market['future']) {\n            method += 'Future';\n            request['contract_type'] = 'this_week'; // next_week, quarter\n        }\n        method += 'Ticker';\n        let response = await this[method] (this.extend (request, params));\n        let timestamp = parseInt (response['date']) * 1000;\n        let ticker = this.extend (response['ticker'], { 'timestamp': timestamp });\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'info': trade,\n            'timestamp': trade['date_ms'],\n            'datetime': this.iso8601 (trade['date_ms']),\n            'symbol': symbol,\n            'id': trade['tid'].toString (),\n            'order': undefined,\n            'type': undefined,\n            'side': trade['type'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'publicGet';\n        let request = {\n            'symbol': market['id'],\n        };\n        if (market['future']) {\n            method += 'Future';\n            request['contract_type'] = 'this_week'; // next_week, quarter\n        }\n        method += 'Trades';\n        let response = await this[method] (this.extend (request, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = 1440, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'publicGet';\n        let request = {\n            'symbol': market['id'],\n            'type': this.timeframes[timeframe],\n        };\n        if (market['future']) {\n            method += 'Future';\n            request['contract_type'] = 'this_week'; // next_week, quarter\n        }\n        method += 'Kline';\n        if (limit)\n            request['size'] = parseInt (limit);\n        if (since) {\n            request['since'] = since;\n        } else {\n            request['since'] = this.milliseconds () - 86400000; // last 24 hours\n        }\n        let response = await this[method] (this.extend (request, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostUserinfo ();\n        let balances = response['info']['funds'];\n        let result = { 'info': response };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let account = this.account ();\n            account['free'] = this.safeFloat (balances['free'], lowercase, 0.0);\n            account['used'] = this.safeFloat (balances['freezed'], lowercase, 0.0);\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'privatePost';\n        let order = {\n            'symbol': market['id'],\n            'type': side,\n        };\n        if (market['future']) {\n            method += 'Future';\n            order = this.extend (order, {\n                'contract_type': 'this_week', // next_week, quarter\n                'match_price': 0, // match best counter party price? 0 or 1, ignores price if 1\n                'lever_rate': 10, // leverage rate value: 10 or 20 (10 by default)\n                'price': price,\n                'amount': amount,\n            });\n        } else {\n            if (type === 'limit') {\n                order['price'] = price;\n                order['amount'] = amount;\n            } else {\n                order['type'] += '_market';\n                if (side === 'buy') {\n                    order['price'] = this.safeFloat (params, 'cost');\n                    if (!order['price'])\n                        throw new ExchangeError (this.id + ' market buy orders require an additional cost parameter, cost = price * amount');\n                } else {\n                    order['amount'] = amount;\n                }\n            }\n        }\n        params = this.omit (params, 'cost');\n        method += 'Trade';\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['order_id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' cancelOrder() requires a symbol argument');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n            'order_id': id,\n        };\n        let method = 'privatePost';\n        if (market['future']) {\n            method += 'FutureCancel';\n            request['contract_type'] = 'this_week'; // next_week, quarter\n        } else {\n            method += 'CancelOrder';\n        }\n        let response = await this[method] (this.extend (request, params));\n        return response;\n    }\n\n    parseOrderStatus (status) {\n        if (status === -1)\n            return 'canceled';\n        if (status === 0)\n            return 'open';\n        if (status === 1)\n            return 'partial';\n        if (status === 2)\n            return 'closed';\n        if (status === 4)\n            return 'canceled';\n        return status;\n    }\n\n    parseOrder (order, market = undefined) {\n        let side = undefined;\n        let type = undefined;\n        if ('type' in order) {\n            if ((order['type'] === 'buy') || (order['type'] === 'sell')) {\n                side = order['type'];\n                type = 'limit';\n            } else {\n                side = (order['type'] === 'buy_market') ? 'buy' : 'sell';\n                type = 'market';\n            }\n        }\n        let status = this.parseOrderStatus (order['status']);\n        let symbol = undefined;\n        if (!market) {\n            if ('symbol' in order)\n                if (order['symbol'] in this.markets_by_id)\n                    market = this.markets_by_id[order['symbol']];\n        }\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = undefined;\n        let createDateField = this.getCreateDateField ();\n        if (createDateField in order)\n            timestamp = order[createDateField];\n        let amount = order['amount'];\n        let filled = order['deal_amount'];\n        let remaining = amount - filled;\n        let average = order['avg_price'];\n        let cost = average * filled;\n        let result = {\n            'info': order,\n            'id': order['order_id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': order['price'],\n            'average': average,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': undefined,\n        };\n        return result;\n    }\n\n    getCreateDateField () {\n        // needed for derived exchanges\n        // allcoin typo create_data instead of create_date\n        return 'create_date';\n    }\n\n    getOrdersField () {\n        // needed for derived exchanges\n        // allcoin typo order instead of orders (expected based on their API docs)\n        return 'orders';\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrder requires a symbol parameter');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'privatePost';\n        let request = {\n            'order_id': id,\n            'symbol': market['id'],\n            // 'status': 0, // 0 for unfilled orders, 1 for filled orders\n            // 'current_page': 1, // current page number\n            // 'page_length': 200, // number of orders returned per page, maximum 200\n        };\n        if (market['future']) {\n            method += 'Future';\n            request['contract_type'] = 'this_week'; // next_week, quarter\n        }\n        method += 'OrderInfo';\n        let response = await this[method] (this.extend (request, params));\n        let ordersField = this.getOrdersField ();\n        if (response[ordersField].length > 0)\n            return this.parseOrder (response[ordersField][0]);\n        throw new OrderNotFound (this.id + ' order ' + id + ' not found');\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrders requires a symbol parameter');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'privatePost';\n        let request = {\n            'symbol': market['id'],\n        };\n        let order_id_in_params = ('order_id' in params);\n        if (market['future']) {\n            method += 'FutureOrdersInfo';\n            request['contract_type'] = 'this_week'; // next_week, quarter\n            if (!order_id_in_params)\n                throw new ExchangeError (this.id + ' fetchOrders() requires order_id param for futures market ' + symbol + ' (a string of one or more order ids, comma-separated)');\n        } else {\n            let status = undefined;\n            if ('type' in params) {\n                status = params['type'];\n            } else if ('status' in params) {\n                status = params['status'];\n            } else {\n                let name = order_id_in_params ? 'type' : 'status';\n                throw new ExchangeError (this.id + ' fetchOrders() requires ' + name + ' param for spot market ' + symbol + ' (0 - for unfilled orders, 1 - for filled/canceled orders)');\n            }\n            if (order_id_in_params) {\n                method += 'OrdersInfo';\n                request = this.extend (request, {\n                    'type': status,\n                });\n            } else {\n                method += 'OrderHistory';\n                request = this.extend (request, {\n                    'status': status,\n                    'current_page': 1, // current page number\n                    'page_length': 200, // number of orders returned per page, maximum 200\n                });\n            }\n            params = this.omit (params, [ 'type', 'status' ]);\n        }\n        let response = await this[method] (this.extend (request, params));\n        let ordersField = this.getOrdersField ();\n        return this.parseOrders (response[ordersField], market, since, limit);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let open = 0; // 0 for unfilled orders, 1 for filled orders\n        return await this.fetchOrders (symbol, undefined, undefined, this.extend ({\n            'status': open,\n        }, params));\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let closed = 1; // 0 for unfilled orders, 1 for filled orders\n        let orders = await this.fetchOrders (symbol, undefined, undefined, this.extend ({\n            'status': closed,\n        }, params));\n        return this.filterBy (orders, 'status', 'closed');\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let lowercase = currency.toLowerCase () + '_usd';\n        // if (amount < 0.01)\n        //     throw new ExchangeError (this.id + ' withdraw() requires amount > 0.01');\n        let request = {\n            'symbol': lowercase,\n            'withdraw_address': address,\n            'withdraw_amount': amount,\n            'target': 'address', // or okcn, okcom, okex\n        };\n        let query = params;\n        if ('chargefee' in query) {\n            request['chargefee'] = query['chargefee'];\n            query = this.omit (query, 'chargefee');\n        } else {\n            throw new ExchangeError (this.id + ' withdraw() requires a `chargefee` parameter');\n        }\n        let password = undefined;\n        if (this.password) {\n            request['trade_pwd'] = this.password;\n            password = this.password;\n        } else if ('password' in query) {\n            request['trade_pwd'] = query['password'];\n            query = this.omit (query, 'password');\n        } else if ('trade_pwd' in query) {\n            request['trade_pwd'] = query['trade_pwd'];\n            query = this.omit (query, 'trade_pwd');\n        }\n        if (!password)\n            throw new ExchangeError (this.id + ' withdraw() requires this.password set on the exchange instance or a password / trade_pwd parameter');\n        let response = await this.privatePostWithdraw (this.extend (request, query));\n        return {\n            'info': response,\n            'id': this.safeString (response, 'withdraw_id'),\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = '/';\n        if (api !== 'web')\n            url += this.version + '/';\n        url += path + this.extension;\n        if (api === 'private') {\n            this.checkRequiredCredentials ();\n            let query = this.keysort (this.extend ({\n                'api_key': this.apiKey,\n            }, params));\n            // secret key must be at the end of query\n            let queryString = this.rawencode (query) + '&secret_key=' + this.secret;\n            query['sign'] = this.hash (this.encode (queryString)).toUpperCase ();\n            body = this.urlencode (query);\n            headers = { 'Content-Type': 'application/x-www-form-urlencoded' };\n        } else {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        }\n        url = this.urls['api'][api] + url;\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (body.length < 2)\n            return; // fallback to default error handler\n        if (body[0] === '{') {\n            let response = JSON.parse (body);\n            if ('error_code' in response) {\n                let error = this.safeString (response, 'error_code');\n                let message = this.id + ' ' + this.json (response);\n                if (error in this.exceptions) {\n                    let ExceptionClass = this.exceptions[error];\n                    throw new ExceptionClass (message);\n                } else {\n                    throw new ExchangeError (message);\n                }\n            }\n            if ('result' in response)\n                if (!response['result'])\n                    throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n    }\n};\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst okcoinusd = require ('./okcoinusd.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class okex extends okcoinusd {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'okex',\n            'name': 'OKEX',\n            'countries': [ 'CN', 'US' ],\n            'hasCORS': false,\n            'hasFutureMarkets': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/32552768-0d6dd3c6-c4a6-11e7-90f8-c043b64756a7.jpg',\n                'api': {\n                    'web': 'https://www.okex.com/v2',\n                    'public': 'https://www.okex.com/api',\n                    'private': 'https://www.okex.com/api',\n                },\n                'www': 'https://www.okex.com',\n                'doc': 'https://www.okex.com/rest_getStarted.html',\n                'fees': 'https://www.okex.com/fees.html',\n            },\n        });\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class paymium extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'paymium',\n            'name': 'Paymium',\n            'countries': [ 'FR', 'EU' ],\n            'rateLimit': 2000,\n            'version': 'v1',\n            'hasCORS': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27790564-a945a9d4-5ff9-11e7-9d2d-b635763f2f24.jpg',\n                'api': 'https://paymium.com/api',\n                'www': 'https://www.paymium.com',\n                'doc': [\n                    'https://github.com/Paymium/api-documentation',\n                    'https://www.paymium.com/page/developers',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'countries',\n                        'data/{id}/ticker',\n                        'data/{id}/trades',\n                        'data/{id}/depth',\n                        'bitcoin_charts/{id}/trades',\n                        'bitcoin_charts/{id}/depth',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'merchant/get_payment/{UUID}',\n                        'user',\n                        'user/addresses',\n                        'user/addresses/{btc_address}',\n                        'user/orders',\n                        'user/orders/{UUID}',\n                        'user/price_alerts',\n                    ],\n                    'post': [\n                        'user/orders',\n                        'user/addresses',\n                        'user/payment_requests',\n                        'user/price_alerts',\n                        'merchant/create_payment',\n                    ],\n                    'delete': [\n                        'user/orders/{UUID}/cancel',\n                        'user/price_alerts/{id}',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/EUR': { 'id': 'eur', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR' },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.0059,\n                    'taker': 0.0059,\n                },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let balances = await this.privateGetUser ();\n        let result = { 'info': balances };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let account = this.account ();\n            let balance = 'balance_' + lowercase;\n            let locked = 'locked_' + lowercase;\n            if (balance in balances)\n                account['free'] = balances[balance];\n            if (locked in balances)\n                account['used'] = balances[locked];\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetDataIdDepth (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        let result = this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'amount');\n        result['bids'] = this.sortBy (result['bids'], 0, true);\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetDataIdTicker (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        let timestamp = ticker['at'] * 1000;\n        let vwap = parseFloat (ticker['vwap']);\n        let baseVolume = parseFloat (ticker['volume']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'bid'),\n            'ask': this.safeFloat (ticker, 'ask'),\n            'vwap': vwap,\n            'open': this.safeFloat (ticker, 'open'),\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'price'),\n            'change': undefined,\n            'percentage': this.safeFloat (ticker, 'variation'),\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['created_at_int']) * 1000;\n        let volume = 'traded_' + market['base'].toLowerCase ();\n        return {\n            'info': trade,\n            'id': trade['uuid'],\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['side'],\n            'price': trade['price'],\n            'amount': trade[volume],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetDataIdTrades (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (market, type, side, amount, price = undefined, params = {}) {\n        let order = {\n            'type': this.capitalize (type) + 'Order',\n            'currency': this.marketId (market),\n            'direction': side,\n            'amount': amount,\n        };\n        if (type == 'market')\n            order['price'] = price;\n        let response = await this.privatePostUserOrders (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['uuid'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelOrder (this.extend ({\n            'orderNumber': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            body = this.json (params);\n            let nonce = this.nonce ().toString ();\n            let auth = nonce + url + body;\n            headers = {\n                'Api-Key': this.apiKey,\n                'Api-Signature': this.hmac (this.encode (auth), this.secret),\n                'Api-Nonce': nonce,\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('errors' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeNotAvailable, ExchangeError, InsufficientFunds, OrderNotFound, OrderNotCached, InvalidOrder } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class poloniex extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'poloniex',\n            'name': 'Poloniex',\n            'countries': 'US',\n            'rateLimit': 1000, // up to 6 calls per second\n            'hasCORS': true,\n            // obsolete metainfo interface\n            'hasFetchMyTrades': true,\n            'hasFetchOrder': true,\n            'hasFetchOrders': true,\n            'hasFetchOpenOrders': true,\n            'hasFetchClosedOrders': true,\n            'hasFetchTickers': true,\n            'hasFetchCurrencies': true,\n            'hasWithdraw': true,\n            'hasFetchOHLCV': true,\n            // new metainfo interface\n            'has': {\n                'fetchOHLCV': true,\n                'fetchMyTrades': true,\n                'fetchOrder': 'emulated',\n                'fetchOrders': 'emulated',\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': 'emulated',\n                'fetchTickers': true,\n                'fetchCurrencies': true,\n                'withdraw': true,\n            },\n            'timeframes': {\n                '5m': 300,\n                '15m': 900,\n                '30m': 1800,\n                '2h': 7200,\n                '4h': 14400,\n                '1d': 86400,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766817-e9456312-5ee6-11e7-9b3c-b628ca5626a5.jpg',\n                'api': {\n                    'public': 'https://poloniex.com/public',\n                    'private': 'https://poloniex.com/tradingApi',\n                },\n                'www': 'https://poloniex.com',\n                'doc': [\n                    'https://poloniex.com/support/api/',\n                    'http://pastebin.com/dMX7mZE0',\n                ],\n                'fees': 'https://poloniex.com/fees',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'return24hVolume',\n                        'returnChartData',\n                        'returnCurrencies',\n                        'returnLoanOrders',\n                        'returnOrderBook',\n                        'returnTicker',\n                        'returnTradeHistory',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'buy',\n                        'cancelLoanOffer',\n                        'cancelOrder',\n                        'closeMarginPosition',\n                        'createLoanOffer',\n                        'generateNewAddress',\n                        'getMarginPosition',\n                        'marginBuy',\n                        'marginSell',\n                        'moveOrder',\n                        'returnActiveLoans',\n                        'returnAvailableAccountBalances',\n                        'returnBalances',\n                        'returnCompleteBalances',\n                        'returnDepositAddresses',\n                        'returnDepositsWithdrawals',\n                        'returnFeeInfo',\n                        'returnLendingHistory',\n                        'returnMarginAccountSummary',\n                        'returnOpenLoanOffers',\n                        'returnOpenOrders',\n                        'returnOrderTrades',\n                        'returnTradableBalances',\n                        'returnTradeHistory',\n                        'sell',\n                        'toggleAutoRenew',\n                        'transferBalance',\n                        'withdraw',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.0015,\n                    'taker': 0.0025,\n                },\n                'funding': 0.0,\n            },\n            'limits': {\n                'amount': {\n                    'min': 0.00000001,\n                    'max': 1000000000,\n                },\n                'price': {\n                    'min': 0.00000001,\n                    'max': 1000000000,\n                },\n                'cost': {\n                    'min': 0.00000000,\n                    'max': 1000000000,\n                },\n            },\n            'precision': {\n                'amount': 8,\n                'price': 8,\n            },\n        });\n    }\n\n    calculateFee (symbol, type, side, amount, price, takerOrMaker = 'taker', params = {}) {\n        let market = this.markets[symbol];\n        let key = 'quote';\n        let rate = market[takerOrMaker];\n        let cost = parseFloat (this.costToPrecision (symbol, amount * rate));\n        if (side === 'sell') {\n            cost *= price;\n        } else {\n            key = 'base';\n        }\n        return {\n            'type': takerOrMaker,\n            'currency': market[key],\n            'rate': rate,\n            'cost': parseFloat (this.feeToPrecision (symbol, cost)),\n        };\n    }\n\n    commonCurrencyCode (currency) {\n        if (currency === 'BTM')\n            return 'Bitmark';\n        if (currency === 'STR')\n            return 'XLM';\n        return currency;\n    }\n\n    currencyId (currency) {\n        if (currency === 'Bitmark')\n            return 'BTM';\n        if (currency === 'XLM')\n            return 'STR';\n        return currency;\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '5m', since = undefined, limit = undefined) {\n        return [\n            ohlcv['date'] * 1000,\n            ohlcv['open'],\n            ohlcv['high'],\n            ohlcv['low'],\n            ohlcv['close'],\n            ohlcv['volume'],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '5m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        if (!since)\n            since = 0;\n        let request = {\n            'currencyPair': market['id'],\n            'period': this.timeframes[timeframe],\n            'start': parseInt (since / 1000),\n        };\n        if (limit)\n            request['end'] = this.sum (request['start'], limit * this.timeframes[timeframe]);\n        let response = await this.publicGetReturnChartData (this.extend (request, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetReturnTicker ();\n        let keys = Object.keys (markets);\n        let result = [];\n        for (let p = 0; p < keys.length; p++) {\n            let id = keys[p];\n            let market = markets[id];\n            let [ quote, base ] = id.split ('_');\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': true,\n                'lot': this.limits['amount']['min'],\n                'info': market,\n            }));\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privatePostReturnCompleteBalances (this.extend ({\n            'account': 'all',\n        }, params));\n        let result = { 'info': balances };\n        let currencies = Object.keys (balances);\n        for (let c = 0; c < currencies.length; c++) {\n            let id = currencies[c];\n            let balance = balances[id];\n            let currency = this.commonCurrencyCode (id);\n            let account = {\n                'free': parseFloat (balance['available']),\n                'used': parseFloat (balance['onOrders']),\n                'total': 0.0,\n            };\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchFees (params = {}) {\n        await this.loadMarkets ();\n        let fees = await this.privatePostReturnFeeInfo ();\n        return {\n            'info': fees,\n            'maker': parseFloat (fees['makerFee']),\n            'taker': parseFloat (fees['takerFee']),\n            'withdraw': 0.0,\n        };\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetReturnOrderBook (this.extend ({\n            'currencyPair': this.marketId (symbol),\n            // 'depth': 100,\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high24hr']),\n            'low': parseFloat (ticker['low24hr']),\n            'bid': parseFloat (ticker['highestBid']),\n            'ask': parseFloat (ticker['lowestAsk']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': parseFloat (ticker['percentChange']),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['quoteVolume']),\n            'quoteVolume': parseFloat (ticker['baseVolume']),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetReturnTicker (params);\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchCurrencies (params = {}) {\n        let currencies = await this.publicGetReturnCurrencies (params);\n        let ids = Object.keys (currencies);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let currency = currencies[id];\n            // todo: will need to rethink the fees\n            // to add support for multiple withdrawal/deposit methods and\n            // differentiated fees for each particular method\n            let precision = 8; // default precision, todo: fix \"magic constants\"\n            let code = this.commonCurrencyCode (id);\n            let active = (currency['delisted'] === 0);\n            let status = (currency['disabled']) ? 'disabled' : 'ok';\n            if (status !== 'ok')\n                active = false;\n            result[code] = {\n                'id': id,\n                'code': code,\n                'info': currency,\n                'name': currency['name'],\n                'active': active,\n                'status': status,\n                'fee': currency['txFee'], // todo: redesign\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': currency['txFee'],\n                        'max': Math.pow (10, precision),\n                    },\n                },\n            };\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let tickers = await this.publicGetReturnTicker (params);\n        let ticker = tickers[market['id']];\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = this.parse8601 (trade['date']);\n        let symbol = undefined;\n        let base = undefined;\n        let quote = undefined;\n        if ((!market) && ('currencyPair' in trade)) {\n            let currencyPair = trade['currencyPair'];\n            if (currencyPair in this.markets_by_id) {\n                market = this.markets_by_id[currencyPair];\n            } else {\n                let parts = currencyPair.split ('_');\n                quote = parts[0];\n                base = parts[1];\n                symbol = base + '/' + quote;\n            }\n        }\n        if (market) {\n            symbol = market['symbol'];\n            base = market['base'];\n            quote = market['quote'];\n        }\n        let side = trade['type'];\n        let fee = undefined;\n        let cost = this.safeFloat (trade, 'total');\n        let amount = parseFloat (trade['amount']);\n        if ('fee' in trade) {\n            let rate = parseFloat (trade['fee']);\n            let feeCost = undefined;\n            let currency = undefined;\n            if (side === 'buy') {\n                currency = base;\n                feeCost = amount * rate;\n            } else {\n                currency = quote;\n                if (typeof cost !== 'undefined')\n                    feeCost = cost * rate;\n            }\n            fee = {\n                'type': undefined,\n                'rate': rate,\n                'cost': feeCost,\n                'currency': currency,\n            };\n        }\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'id': this.safeString (trade, 'tradeID'),\n            'order': this.safeString (trade, 'orderNumber'),\n            'type': 'limit',\n            'side': side,\n            'price': parseFloat (trade['rate']),\n            'amount': amount,\n            'cost': cost,\n            'fee': fee,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'currencyPair': market['id'],\n        };\n        if (since) {\n            request['start'] = parseInt (since / 1000);\n            request['end'] = this.seconds (); // last 50000 trades by default\n        }\n        let trades = await this.publicGetReturnTradeHistory (this.extend (request, params));\n        return this.parseTrades (trades, market, since, limit);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        if (symbol)\n            market = this.market (symbol);\n        let pair = market ? market['id'] : 'all';\n        let request = { 'currencyPair': pair };\n        if (since) {\n            request['start'] = parseInt (since / 1000);\n            request['end'] = this.seconds ();\n        }\n        // limit is disabled (does not really work as expected)\n        // if (limit)\n        //     request['limit'] = parseInt (limit);\n        let response = await this.privatePostReturnTradeHistory (this.extend (request, params));\n        let result = [];\n        if (market) {\n            result = this.parseTrades (response, market);\n        } else {\n            if (response) {\n                let ids = Object.keys (response);\n                for (let i = 0; i < ids.length; i++) {\n                    let id = ids[i];\n                    let market = undefined;\n                    if (id in this.markets_by_id)\n                        market = this.markets_by_id[id];\n                    let trades = this.parseTrades (response[id], market);\n                    for (let j = 0; j < trades.length; j++) {\n                        result.push (trades[j]);\n                    }\n                }\n            }\n        }\n        return this.filterBySinceLimit (result, since, limit);\n    }\n\n    parseOrder (order, market = undefined) {\n        let timestamp = this.safeInteger (order, 'timestamp');\n        if (!timestamp)\n            timestamp = this.parse8601 (order['date']);\n        let trades = undefined;\n        if ('resultingTrades' in order)\n            trades = this.parseTrades (order['resultingTrades'], market);\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let price = this.safeFloat (order, 'price');\n        let cost = this.safeFloat (order, 'total', 0.0);\n        let remaining = this.safeFloat (order, 'amount');\n        let amount = this.safeFloat (order, 'startingAmount', remaining);\n        let filled = undefined;\n        if (typeof amount !== 'undefined') {\n            if (typeof remaining !== 'undefined')\n                filled = amount - remaining;\n        }\n        if (typeof filled === 'undefined') {\n            if (typeof trades !== 'undefined') {\n                filled = 0;\n                cost = 0;\n                for (let i = 0; i < trades.length; i++) {\n                    let trade = trades[i];\n                    let tradeAmount = trade['amount'];\n                    let tradePrice = trade['price'];\n                    filled = this.sum (filled, tradeAmount);\n                    cost += tradePrice * tradeAmount;\n                }\n            }\n        }\n        return {\n            'info': order,\n            'id': order['orderNumber'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': order['status'],\n            'symbol': symbol,\n            'type': order['type'],\n            'side': order['side'],\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'trades': trades,\n            'fee': undefined,\n        };\n    }\n\n    parseOpenOrders (orders, market, result = []) {\n        for (let i = 0; i < orders.length; i++) {\n            let order = orders[i];\n            let extended = this.extend (order, {\n                'status': 'open',\n                'type': 'limit',\n                'side': order['type'],\n                'price': order['rate'],\n            });\n            result.push (this.parseOrder (extended, market));\n        }\n        return result;\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        if (symbol)\n            market = this.market (symbol);\n        let pair = market ? market['id'] : 'all';\n        let response = await this.privatePostReturnOpenOrders (this.extend ({\n            'currencyPair': pair,\n        }));\n        let openOrders = [];\n        if (market) {\n            openOrders = this.parseOpenOrders (response, market, openOrders);\n        } else {\n            let marketIds = Object.keys (response);\n            for (let i = 0; i < marketIds.length; i++) {\n                let marketId = marketIds[i];\n                let orders = response[marketId];\n                let m = this.markets_by_id[marketId];\n                openOrders = this.parseOpenOrders (orders, m, openOrders);\n            }\n        }\n        for (let j = 0; j < openOrders.length; j++) {\n            this.orders[openOrders[j]['id']] = openOrders[j];\n        }\n        let openOrdersIndexedById = this.indexBy (openOrders, 'id');\n        let cachedOrderIds = Object.keys (this.orders);\n        let result = [];\n        for (let k = 0; k < cachedOrderIds.length; k++) {\n            let id = cachedOrderIds[k];\n            if (id in openOrdersIndexedById) {\n                this.orders[id] = this.extend (this.orders[id], openOrdersIndexedById[id]);\n            } else {\n                let order = this.orders[id];\n                if (order['status'] === 'open') {\n                    this.orders[id] = this.extend (order, {\n                        'status': 'closed',\n                        'cost': order['amount'] * order['price'],\n                        'filled': order['amount'],\n                        'remaining': 0.0,\n                    });\n                }\n            }\n            let order = this.orders[id];\n            if (market) {\n                if (order['symbol'] === symbol)\n                    result.push (order);\n            } else {\n                result.push (order);\n            }\n        }\n        return this.filterBySinceLimit (result, since, limit);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        let since = this.safeValue (params, 'since');\n        let limit = this.safeValue (params, 'limit');\n        let request = this.omit (params, [ 'since', 'limit' ]);\n        let orders = await this.fetchOrders (symbol, since, limit, request);\n        for (let i = 0; i < orders.length; i++) {\n            if (orders[i]['id'] === id)\n                return orders[i];\n        }\n        throw new OrderNotCached (this.id + ' order id ' + id.toString () + ' not found in cache');\n    }\n\n    filterOrdersByStatus (orders, status) {\n        let result = [];\n        for (let i = 0; i < orders.length; i++) {\n            if (orders[i]['status'] === status)\n                result.push (orders[i]);\n        }\n        return result;\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, since, limit, params);\n        return this.filterOrdersByStatus (orders, 'open');\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, since, limit, params);\n        return this.filterOrdersByStatus (orders, 'closed');\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type === 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        await this.loadMarkets ();\n        let method = 'privatePost' + this.capitalize (side);\n        let market = this.market (symbol);\n        price = parseFloat (price);\n        amount = parseFloat (amount);\n        let response = await this[method] (this.extend ({\n            'currencyPair': market['id'],\n            'rate': this.priceToPrecision (symbol, price),\n            'amount': this.amountToPrecision (symbol, amount),\n        }, params));\n        let timestamp = this.milliseconds ();\n        let order = this.parseOrder (this.extend ({\n            'timestamp': timestamp,\n            'status': 'open',\n            'type': type,\n            'side': side,\n            'price': price,\n            'amount': amount,\n        }, response), market);\n        let id = order['id'];\n        this.orders[id] = order;\n        return this.extend ({ 'info': response }, order);\n    }\n\n    async editOrder (id, symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        price = parseFloat (price);\n        let request = {\n            'orderNumber': id,\n            'rate': this.priceToPrecision (symbol, price),\n        };\n        if (typeof amount !== 'undefined') {\n            amount = parseFloat (amount);\n            request['amount'] = this.amountToPrecision (symbol, amount);\n        }\n        let response = await this.privatePostMoveOrder (this.extend (request, params));\n        let result = undefined;\n        if (id in this.orders) {\n            this.orders[id]['status'] = 'canceled';\n            let newid = response['orderNumber'];\n            this.orders[newid] = this.extend (this.orders[id], {\n                'id': newid,\n                'price': price,\n                'status': 'open',\n            });\n            if (typeof amount !== 'undefined')\n                this.orders[newid]['amount'] = amount;\n            result = this.extend (this.orders[newid], { 'info': response });\n        } else {\n            let market = undefined;\n            if (symbol)\n                market = this.market (symbol);\n            result = this.parseOrder (response, market);\n            this.orders[result['id']] = result;\n        }\n        return result;\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = undefined;\n        try {\n            response = await this.privatePostCancelOrder (this.extend ({\n                'orderNumber': id,\n            }, params));\n            if (id in this.orders)\n                this.orders[id]['status'] = 'canceled';\n        } catch (e) {\n            if (this.last_http_response) {\n                if (this.last_http_response.indexOf ('Invalid order') >= 0)\n                    throw new OrderNotFound (this.id + ' cancelOrder() error: ' + this.last_http_response);\n            }\n            throw e;\n        }\n        return response;\n    }\n\n    async fetchOrderStatus (id, symbol = undefined) {\n        await this.loadMarkets ();\n        let orders = await this.fetchOpenOrders (symbol);\n        let indexed = this.indexBy (orders, 'id');\n        return (id in indexed) ? 'open' : 'closed';\n    }\n\n    async fetchOrderTrades (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let trades = await this.privatePostReturnOrderTrades (this.extend ({\n            'orderNumber': id,\n        }, params));\n        return this.parseTrades (trades);\n    }\n\n    async createDepositAddress (currency, params = {}) {\n        let currencyId = this.currencyId (currency);\n        let response = await this.privatePostGenerateNewAddress ({\n            'currency': currencyId,\n        });\n        let address = undefined;\n        if (response['success'] === 1)\n            address = this.safeString (response, 'response');\n        if (!address)\n            throw new ExchangeError (this.id + ' createDepositAddress failed: ' + this.last_http_response);\n        return {\n            'currency': currency,\n            'address': address,\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        let response = await this.privatePostReturnDepositAddresses ();\n        let currencyId = this.currencyId (currency);\n        let address = this.safeString (response, currencyId);\n        let status = address ? 'ok' : 'none';\n        return {\n            'currency': currency,\n            'address': address,\n            'status': status,\n            'info': response,\n        };\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let currencyId = this.currencyId (currency);\n        let request = {\n            'currency': currencyId,\n            'amount': amount,\n            'address': address,\n        };\n        if (tag)\n            request['paymentId'] = tag;\n        let result = await this.privatePostWithdraw (this.extend (request, params));\n        return {\n            'info': result,\n            'id': result['response'],\n        };\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        let query = this.extend ({ 'command': path }, params);\n        if (api === 'public') {\n            url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            query['nonce'] = this.nonce ();\n            body = this.urlencode (query);\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Key': this.apiKey,\n                'Sign': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code >= 400) {\n            if (body[0] === '{') {\n                let response = JSON.parse (body);\n                if ('error' in response) {\n                    let error = this.id + ' ' + body;\n                    if (response['error'].indexOf ('Total must be at least') >= 0) {\n                        throw new InvalidOrder (error);\n                    } else if (response['error'].indexOf ('Not enough') >= 0) {\n                        throw new InsufficientFunds (error);\n                    } else if (response['error'].indexOf ('Nonce must be greater') >= 0) {\n                        throw new ExchangeNotAvailable (error);\n                    }\n                }\n            }\n        }\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response) {\n            let error = this.id + ' ' + this.json (response);\n            let failed = response['error'].indexOf ('Not enough') >= 0;\n            if (failed)\n                throw new InsufficientFunds (error);\n            throw new ExchangeError (error);\n        }\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, OrderNotFound, InvalidOrder, InsufficientFunds, AuthenticationError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class qryptos extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'qryptos',\n            'name': 'QRYPTOS',\n            'countries': [ 'CN', 'TW' ],\n            'version': '2',\n            'rateLimit': 1000,\n            'hasFetchTickers': true,\n            'hasCORS': false,\n            'has': {\n                'fetchOrder': true,\n                'fetchOrders': true,\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30953915-b1611dc0-a436-11e7-8947-c95bd5a42086.jpg',\n                'api': 'https://api.qryptos.com',\n                'www': 'https://www.qryptos.com',\n                'doc': [\n                    'https://developers.quoine.com',\n                    'https://developers.quoine.com/v2',\n                ],\n                'fees': 'https://qryptos.zendesk.com/hc/en-us/articles/115007858167-Fees',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'products',\n                        'products/{id}',\n                        'products/{id}/price_levels',\n                        'executions',\n                        'ir_ladders/{currency}',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'accounts/balance',\n                        'accounts/main_asset',\n                        'crypto_accounts',\n                        'executions/me',\n                        'fiat_accounts',\n                        'loan_bids',\n                        'loans',\n                        'orders',\n                        'orders/{id}',\n                        'orders/{id}/trades',\n                        'orders/{id}/executions',\n                        'trades',\n                        'trades/{id}/loans',\n                        'trading_accounts',\n                        'trading_accounts/{id}',\n                    ],\n                    'post': [\n                        'fiat_accounts',\n                        'loan_bids',\n                        'orders',\n                    ],\n                    'put': [\n                        'loan_bids/{id}/close',\n                        'loans/{id}',\n                        'orders/{id}',\n                        'orders/{id}/cancel',\n                        'trades/{id}',\n                        'trades/{id}/close',\n                        'trades/close_all',\n                        'trading_accounts/{id}',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetProducts ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['id'].toString ();\n            let base = market['base_currency'];\n            let quote = market['quoted_currency'];\n            let symbol = base + '/' + quote;\n            let maker = this.safeFloat (market, 'maker_fee');\n            let taker = this.safeFloat (market, 'taker_fee');\n            let active = !market['disabled'];\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'maker': maker,\n                'taker': taker,\n                'active': active,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privateGetAccountsBalance ();\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let total = parseFloat (balance['balance']);\n            let account = {\n                'free': total,\n                'used': 0.0,\n                'total': total,\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetProductsIdPriceLevels (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'buy_price_levels', 'sell_price_levels');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let last = undefined;\n        if ('last_traded_price' in ticker) {\n            if (ticker['last_traded_price']) {\n                let length = ticker['last_traded_price'].length;\n                if (length > 0)\n                    last = parseFloat (ticker['last_traded_price']);\n            }\n        }\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high_market_ask'),\n            'low': this.safeFloat (ticker, 'low_market_bid'),\n            'bid': this.safeFloat (ticker, 'market_bid'),\n            'ask': this.safeFloat (ticker, 'market_ask'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': last,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'volume_24h'),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetProducts (params);\n        let result = {};\n        for (let t = 0; t < tickers.length; t++) {\n            let ticker = tickers[t];\n            let base = ticker['base_currency'];\n            let quote = ticker['quoted_currency'];\n            let symbol = base + '/' + quote;\n            let market = this.markets[symbol];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetProductsId (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['created_at'] * 1000;\n        return {\n            'info': trade,\n            'id': trade['id'].toString (),\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['taker_side'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['quantity']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'product_id': market['id'],\n        };\n        if (limit)\n            request['limit'] = limit;\n        let response = await this.publicGetExecutions (this.extend (request, params));\n        return this.parseTrades (response['models'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'order_type': type,\n            'product_id': this.marketId (symbol),\n            'side': side,\n            'quantity': amount,\n        };\n        if (type === 'limit')\n            order['price'] = price;\n        let response = await this.privatePostOrders (this.extend ({\n            'order': order,\n        }, params));\n        return this.parseOrder(response);\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let result = await this.privatePutOrdersIdCancel (this.extend ({\n            'id': id,\n        }, params));\n        let order = this.parseOrder (result);\n        if (order['status'] === 'closed')\n            throw new OrderNotFound (this.id + ' ' + this.json (order));\n        return order;\n    }\n\n    parseOrder (order) {\n        let timestamp = order['created_at'] * 1000;\n        let marketId = order['product_id'].toString ();\n        let market = this.marketsById[marketId];\n        let status = undefined;\n        if ('status' in order) {\n            if (order['status'] === 'live') {\n                status = 'open';\n            } else if (order['status'] === 'filled') {\n                status = 'closed';\n            } else if (order['status'] === 'cancelled') { // 'll' intended\n                status = 'canceled';\n            }\n        }\n        let amount = parseFloat (order['quantity']);\n        let filled = parseFloat (order['filled_quantity']);\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        }\n        return {\n            'id': order['id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'type': order['order_type'],\n            'status': status,\n            'symbol': symbol,\n            'side': order['side'],\n            'price': order['price'],\n            'amount': amount,\n            'filled': filled,\n            'remaining': amount - filled,\n            'trades': undefined,\n            'fee': {\n                'currency': undefined,\n                'cost': parseFloat (order['order_fee']),\n            },\n            'info': order,\n        };\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = await this.privateGetOrdersId (this.extend ({\n            'id': id,\n        }, params));\n        return this.parseOrder (order);\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params={}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        let request = {};\n        if (symbol) {\n            market = this.market (symbol);\n            request['product_id'] = market['id'];\n        }\n        let status = params['status'];\n        if (status === 'open') {\n            request['status'] = 'live';\n        } else if (status === 'closed') {\n            request['status'] = 'filled';\n        } else if (status === 'canceled') {\n            request['status'] = 'cancelled';\n        }\n        let result = await this.privateGetOrders (request);\n        let orders = result['models'];\n        return this.parseOrders (orders, market, since, limit);\n    }\n\n    fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        return this.fetchOrders (symbol, since, limit, this.extend ({ 'status': 'open' }, params));\n    }\n\n    fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        return this.fetchOrders (symbol, since, limit, this.extend ({ 'status': 'closed' }, params));\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        let response = undefined;\n        if (code === 200 || code === 404 || code === 422) {\n            if ((body[0] === '{') || (body[0] === '[')) {\n                response = JSON.parse (body);\n            } else {\n                // if not a JSON response\n                throw new ExchangeError (this.id + ' returned a non-JSON reply: ' + body);\n            }\n        }\n        if (code === 401) {\n            if (body === 'API Authentication failed') {\n                throw new AuthenticationError (body);\n            }\n        }\n        if (code === 404) {\n            if ('message' in response) {\n                if (response['message'] === 'Order not found') {\n                    throw new OrderNotFound (this.id + ' ' + body);\n                }\n            }\n        } else if (code === 422) {\n            if ('errors' in response) {\n                let errors = response['errors'];\n                if ('user' in errors) {\n                    let messages = errors['user'];\n                    if (messages.indexOf ('not_enough_free_balance') >= 0) {\n                        throw new InsufficientFunds (this.id + ' ' + body);\n                    }\n                } else if ('quantity' in errors) {\n                    let messages = errors['quantity'];\n                    if (messages.indexOf ('less_than_order_size') >= 0) {\n                        throw new InvalidOrder (this.id + ' ' + body);\n                    }\n                }\n            }\n        }\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        headers = {\n            'X-Quoine-API-Version': this.version,\n            'Content-Type': 'application/json',\n        };\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            if (method === 'GET') {\n                if (Object.keys (query).length)\n                    url += '?' + this.urlencode (query);\n            } else if (Object.keys (query).length) {\n                body = this.json (query);\n            }\n            let nonce = this.nonce ();\n            let request = {\n                'path': url,\n                'nonce': nonce,\n                'token_id': this.apiKey,\n                'iat': Math.floor (nonce / 1000), // issued at\n            };\n            headers['X-Quoine-Auth'] = this.jwt (request, this.secret);\n        }\n        url = this.urls['api'] + url;\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n};\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class quadrigacx extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'quadrigacx',\n            'name': 'QuadrigaCX',\n            'countries': 'CA',\n            'rateLimit': 1000,\n            'version': 'v2',\n            'hasCORS': true,\n            // obsolete metainfo interface\n            'hasWithdraw': true,\n            // new metainfo interface\n            'has': {\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766825-98a6d0de-5ee7-11e7-9fa4-38e11a2c6f52.jpg',\n                'api': 'https://api.quadrigacx.com',\n                'www': 'https://www.quadrigacx.com',\n                'doc': 'https://www.quadrigacx.com/api_info',\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'uid': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'order_book',\n                        'ticker',\n                        'transactions',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balance',\n                        'bitcoin_deposit_address',\n                        'bitcoin_withdrawal',\n                        'buy',\n                        'cancel_order',\n                        'ether_deposit_address',\n                        'ether_withdrawal',\n                        'lookup_order',\n                        'open_orders',\n                        'sell',\n                        'user_transactions',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/CAD': { 'id': 'btc_cad', 'symbol': 'BTC/CAD', 'base': 'BTC', 'quote': 'CAD', 'maker': 0.005, 'taker': 0.005 },\n                'BTC/USD': { 'id': 'btc_usd', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD', 'maker': 0.005, 'taker': 0.005 },\n                'ETH/BTC': { 'id': 'eth_btc', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC', 'maker': 0.002, 'taker': 0.002 },\n                'ETH/CAD': { 'id': 'eth_cad', 'symbol': 'ETH/CAD', 'base': 'ETH', 'quote': 'CAD', 'maker': 0.005, 'taker': 0.005 },\n                'LTC/CAD': { 'id': 'ltc_cad', 'symbol': 'LTC/CAD', 'base': 'LTC', 'quote': 'CAD', 'maker': 0.005, 'taker': 0.005 },\n                'BCH/CAD': { 'id': 'bch_cad', 'symbol': 'BCH/CAD', 'base': 'BCH', 'quote': 'CAD', 'maker': 0.005, 'taker': 0.005 },\n                'BTG/CAD': { 'id': 'btg_cad', 'symbol': 'BTG/CAD', 'base': 'BTG', 'quote': 'CAD', 'maker': 0.005, 'taker': 0.005 },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let balances = await this.privatePostBalance ();\n        let result = { 'info': balances };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let account = {\n                'free': parseFloat (balances[lowercase + '_available']),\n                'used': parseFloat (balances[lowercase + '_reserved']),\n                'total': parseFloat (balances[lowercase + '_balance']),\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetOrderBook (this.extend ({\n            'book': this.marketId (symbol),\n        }, params));\n        let timestamp = parseInt (orderbook['timestamp']) * 1000;\n        return this.parseOrderBook (orderbook, timestamp);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetTicker (this.extend ({\n            'book': this.marketId (symbol),\n        }, params));\n        let timestamp = parseInt (ticker['timestamp']) * 1000;\n        let vwap = parseFloat (ticker['vwap']);\n        let baseVolume = parseFloat (ticker['volume']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': vwap,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': trade['tid'].toString (),\n            'order': undefined,\n            'type': undefined,\n            'side': trade['side'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetTransactions (this.extend ({\n            'book': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let method = 'privatePost' + this.capitalize (side);\n        let order = {\n            'amount': amount,\n            'book': this.marketId (symbol),\n        };\n        if (type === 'limit')\n            order['price'] = price;\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelOrder (this.extend ({\n            'id': id,\n        }, params));\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        let method = 'privatePost' + this.getCurrencyName (currency) + 'DepositAddress';\n        let response = await this[method] (params);\n        let address = undefined;\n        let status = undefined;\n        // [E|e]rror\n        if (response.indexOf ('rror') >= 0) {\n            status = 'error';\n        } else {\n            address = response;\n            status = 'ok';\n        }\n        return {\n            'currency': currency,\n            'address': address,\n            'status': status,\n            'info': this.last_http_response,\n        };\n    }\n\n    getCurrencyName (currency) {\n        if (currency === 'ETH')\n            return 'Ether';\n        if (currency === 'BTC')\n            return 'Bitcoin';\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            'amount': amount,\n            'address': address,\n        };\n        let method = 'privatePost' + this.getCurrencyName (currency) + 'Withdrawal';\n        let response = await this[method] (this.extend (request, params));\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + path;\n        if (api === 'public') {\n            url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let request = [ nonce.toString (), this.uid, this.apiKey ].join ('');\n            let signature = this.hmac (this.encode (request), this.encode (this.secret));\n            let query = this.extend ({\n                'key': this.apiKey,\n                'nonce': nonce,\n                'signature': signature,\n            }, params);\n            body = this.json (query);\n            headers = {\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (typeof response === 'string')\n            return response;\n        if ('error' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","'use strict';\n\n// ---------------------------------------------------------------------------\n\nconst qryptos = require ('./qryptos.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class quoinex extends qryptos {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'quoinex',\n            'name': 'QUOINEX',\n            'countries': [ 'JP', 'SG', 'VN' ],\n            'version': '2',\n            'rateLimit': 1000,\n            'hasFetchTickers': true,\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/35047114-0e24ad4a-fbaa-11e7-96a9-69c1a756083b.jpg',\n                'api': 'https://api.quoine.com',\n                'www': 'https://quoinex.com/',\n                'doc': [\n                    'https://developers.quoine.com',\n                    'https://developers.quoine.com/v2',\n                ],\n                'fees': 'https://quoine.zendesk.com/hc/en-us/articles/115011281488-Fees',\n            },\n        });\n    }\n};\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nlet { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class southxchange extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'southxchange',\n            'name': 'SouthXchange',\n            'countries': 'AR', // Argentina\n            'rateLimit': 1000,\n            'hasFetchTickers': true,\n            'hasCORS': false,\n            'hasWithdraw': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27838912-4f94ec8a-60f6-11e7-9e5d-bbf9bd50a559.jpg',\n                'api': 'https://www.southxchange.com/api',\n                'www': 'https://www.southxchange.com',\n                'doc': 'https://www.southxchange.com/Home/Api',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'markets',\n                        'price/{symbol}',\n                        'prices',\n                        'book/{symbol}',\n                        'trades/{symbol}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'cancelMarketOrders',\n                        'cancelOrder',\n                        'generatenewaddress',\n                        'listOrders',\n                        'listBalances',\n                        'placeOrder',\n                        'withdraw',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'maker': 0.2 / 100,\n                    'taker': 0.2 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetMarkets ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let base = market[0];\n            let quote = market[1];\n            let symbol = base + '/' + quote;\n            let id = symbol;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privatePostListBalances ();\n        if (!balances)\n            throw new ExchangeError (this.id + ' fetchBalance got an unrecognized response');\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['Currency'];\n            let uppercase = currency.toUpperCase ();\n            let free = parseFloat (balance['Available']);\n            let used = parseFloat (balance['Unconfirmed']);\n            let total = this.sum (free, used);\n            let account = {\n                'free': free,\n                'used': used,\n                'total': total,\n            };\n            result[uppercase] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetBookSymbol (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'BuyOrders', 'SellOrders', 'Price', 'Amount');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': this.safeFloat (ticker, 'Bid'),\n            'ask': this.safeFloat (ticker, 'Ask'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'Last'),\n            'change': this.safeFloat (ticker, 'Variation24Hr'),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'Volume24Hr'),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetPrices (params);\n        let tickers = this.indexBy (response, 'Market');\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let symbol = id;\n            let market = undefined;\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            }\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetPriceSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['At'] * 1000;\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': undefined,\n            'order': undefined,\n            'type': undefined,\n            'side': trade['Type'],\n            'price': trade['Price'],\n            'amount': trade['Amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTradesSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let order = {\n            'listingCurrency': market['base'],\n            'referenceCurrency': market['quote'],\n            'type': side,\n            'amount': amount,\n        };\n        if (type === 'limit')\n            order['limitPrice'] = price;\n        let response = await this.privatePostPlaceOrder (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response.toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelOrder (this.extend ({\n            'orderCode': id,\n        }, params));\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        let response = await this.privatePostWithdraw (this.extend ({\n            'currency': currency,\n            'address': address,\n            'amount': amount,\n        }, params));\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            query = this.extend ({\n                'key': this.apiKey,\n                'nonce': nonce,\n            }, query);\n            body = this.json (query);\n            headers = {\n                'Content-Type': 'application/json',\n                'Hash': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        return response;\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst foxbit = require ('./foxbit.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class surbitcoin extends foxbit {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'surbitcoin',\n            'name': 'SurBitcoin',\n            'countries': 'VE',\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27991511-f0a50194-6481-11e7-99b5-8f02932424cc.jpg',\n                'api': {\n                    'public': 'https://api.blinktrade.com/api',\n                    'private': 'https://api.blinktrade.com/tapi',\n                },\n                'www': 'https://surbitcoin.com',\n                'doc': 'https://blinktrade.com/docs',\n            },\n        });\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class therock extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'therock',\n            'name': 'TheRockTrading',\n            'countries': 'MT',\n            'rateLimit': 1000,\n            'version': 'v1',\n            'hasCORS': false,\n            'hasFetchTickers': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766869-75057fa2-5ee9-11e7-9a6f-13e641fa4707.jpg',\n                'api': 'https://api.therocktrading.com',\n                'www': 'https://therocktrading.com',\n                'doc': [\n                    'https://api.therocktrading.com/doc/v1/index.html',\n                    'https://api.therocktrading.com/doc/',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'funds/{id}/orderbook',\n                        'funds/{id}/ticker',\n                        'funds/{id}/trades',\n                        'funds/tickers',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'balances',\n                        'balances/{id}',\n                        'discounts',\n                        'discounts/{id}',\n                        'funds',\n                        'funds/{id}',\n                        'funds/{id}/trades',\n                        'funds/{fund_id}/orders',\n                        'funds/{fund_id}/orders/{id}',\n                        'funds/{fund_id}/position_balances',\n                        'funds/{fund_id}/positions',\n                        'funds/{fund_id}/positions/{id}',\n                        'transactions',\n                        'transactions/{id}',\n                        'withdraw_limits/{id}',\n                        'withdraw_limits',\n                    ],\n                    'post': [\n                        'atms/withdraw',\n                        'funds/{fund_id}/orders',\n                    ],\n                    'delete': [\n                        'funds/{fund_id}/orders/{id}',\n                        'funds/{fund_id}/orders/remove_all',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.02 / 100,\n                    'taker': 0.2 / 100,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.0005,\n                        'BCH': 0.0005,\n                        'PPC': 0.02,\n                        'ETH': 0.001,\n                        'ZEC': 0.001,\n                        'LTC': 0.002,\n                        'EUR': 2.5,  // worst-case scenario: https://therocktrading.com/en/pages/fees\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                        'BCH': 0,\n                        'PPC': 0,\n                        'ETH': 0,\n                        'ZEC': 0,\n                        'LTC': 0,\n                        'EUR': 0,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetFundsTickers ();\n        let result = [];\n        for (let p = 0; p < markets['tickers'].length; p++) {\n            let market = markets['tickers'][p];\n            let id = market['fund_id'];\n            let base = id.slice (0, 3);\n            let quote = id.slice (3);\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetBalances ();\n        let balances = response['balances'];\n        let result = { 'info': response };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let free = balance['trading_balance'];\n            let total = balance['balance'];\n            let used = total - free;\n            let account = {\n                'free': free,\n                'used': used,\n                'total': total,\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetFundsIdOrderbook (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        let timestamp = this.parse8601 (orderbook['date']);\n        return this.parseOrderBook (orderbook, timestamp, 'bids', 'asks', 'price', 'amount');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.parse8601 (ticker['date']);\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': parseFloat (ticker['open']),\n            'close': parseFloat (ticker['close']),\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume_traded']),\n            'quoteVolume': parseFloat (ticker['volume']),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetFundsTickers (params);\n        let tickers = this.indexBy (response['tickers'], 'fund_id');\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetFundsIdTicker (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        if (!market)\n            market = this.markets_by_id[trade['fund_id']];\n        let timestamp = this.parse8601 (trade['date']);\n        return {\n            'info': trade,\n            'id': trade['id'].toString (),\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['side'],\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetFundsIdTrades (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTrades (response['trades'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        if (type == 'market')\n            price = 0;\n        let response = await this.privatePostFundsFundIdOrders (this.extend ({\n            'fund_id': this.marketId (symbol),\n            'side': side,\n            'amount': amount,\n            'price': price,\n        }, params));\n        return {\n            'info': response,\n            'id': response['id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privateDeleteFundsFundIdOrdersId (this.extend ({\n            'id': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let auth = nonce + url;\n            headers = {\n                'X-TRT-KEY': this.apiKey,\n                'X-TRT-NONCE': nonce,\n                'X-TRT-SIGN': this.hmac (this.encode (auth), this.encode (this.secret), 'sha512'),\n            };\n            if (Object.keys (query).length) {\n                body = this.json (query);\n                headers['Content-Type'] = 'application/json';\n            }\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('errors' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst liqui = require ('./liqui.js')\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class tidex extends liqui {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'tidex',\n            'name': 'Tidex',\n            'countries': 'UK',\n            'rateLimit': 2000,\n            'version': '3',\n            // 'hasCORS': false,\n            // 'hasFetchTickers': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30781780-03149dc4-a12e-11e7-82bb-313b269d24d4.jpg',\n                'api': {\n                    'public': 'https://api.tidex.com/api',\n                    'private': 'https://api.tidex.com/tapi',\n                },\n                'www': 'https://tidex.com',\n                'doc': 'https://tidex.com/public-api',\n                'fees': [\n                    'https://tidex.com/assets-spec',\n                    'https://tidex.com/pairs-spec',\n                ],\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'taker': 0.1 / 100,\n                    'maker': 0.1 / 100,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.002,\n                        'LTC': 0.001,\n                        'ETH': 0.01,\n                        'DASH': 0.01,\n                        'DOGE': 0.01,\n                        'BTS': 5.0,\n                        'STEEM': 0.01,\n                        'WAVES': 0.01,\n                        'WCT': 0.01,\n                        'WBTC': 0.0001,\n                        'INCNT': 0.1,\n                        'B': 0.1,\n                        'MRT': 1.0,\n                        'MER': 5.0,\n                        'AQUA': 0.001,\n                        'RBX': 100.0,\n                        'TKS': 0.1,\n                        'WUSD': 0.1,\n                        'WEUR': 0.1,\n                        'WGO': 1.0,\n                        'GNT': 1.0,\n                        'EDG': 1.0,\n                        'RLC': 0.3,\n                        'ICN': 0.3,\n                        'WINGS': 1.0,\n                        'VSL': 1.0,\n                        'TIME': 0.01,\n                        'TAAS': 0.3,\n                        'KOLION': 0.1,\n                        'RIDDLE': 10.0,\n                        'ANT': 0.1,\n                        'EFYT': 0.1,\n                        'MGO': 0.5,\n                        'wETT': 1.0,\n                        'eETT': 1.0,\n                        'QRL': 1.0,\n                        'eMGO': 1.0,\n                        'BNT': 1.0,\n                        'SNM': 10.0,\n                        'ZRC': 0.1,\n                        'SNT': 30.0,\n                        'MCO': 0.3,\n                        'STORJ': 1.0,\n                        'EOS': 0.3,\n                        'WGR': 3.0,\n                        'STA': 0.1,\n                        'PBT': 0.0001,\n                        'BCH': 0.00125,\n                        'wSUR': 0.05,\n                        'SUR': 0.05,\n                        'MSP': 5.0,\n                        'InPay': 0.5,\n                        'MTL': 0.1,\n                        'AHT': 0.2,\n                        'PING': 1.0,\n                        'EOT': 0.2,\n                        'AE': 3.0,\n                        'PIX': 10.0,\n                        'CREDO': 30.0,\n                        'LIFE': 1000.0,\n                        'MTH': 5.0,\n                        'BMC': 1.0,\n                        'TRCT': 5.0,\n                        'KNC': 1.0,\n                        'MSD': 0.2,\n                        'SUB': 10.0,\n                        'ENJ': 20.0,\n                        'EVX': 1.0,\n                        'OCL': 3.0,\n                        'ENG': 2.0,\n                        'TDX': 1.0,\n                        'LA': 2.0,\n                        'PRG': 0.5,\n                        'ICOS': 0.03,\n                        'USDT': 40.0,\n                        'ARN': 2.0,\n                        'RYZ': 10.0,\n                        'B2B': 1.0,\n                        'CAT': 10.0,\n                        'SNOV': 25.0,\n                        'DRGN': 3.0,\n                        'TIE': 20.0,\n                        'TRX': 50.0,\n                        'WAX': 5.0,\n                        'AGI': 5.0,\n                        'VEE': 20.0,\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                        'ETH': 0,\n                        'LTC': 0,\n                        'DOGE': 0,\n                        'ICN': 0,\n                        'DASH': 0,\n                        'GNO': 0,\n                        'EOS': 0,\n                        'BCH': 0,\n                        'USDT': 0,\n                    },\n                },\n            },\n        });\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst foxbit = require ('./foxbit.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class urdubit extends foxbit {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'urdubit',\n            'name': 'UrduBit',\n            'countries': 'PK',\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27991453-156bf3ae-6480-11e7-82eb-7295fe1b5bb4.jpg',\n                'api': {\n                    'public': 'https://api.blinktrade.com/api',\n                    'private': 'https://api.blinktrade.com/tapi',\n                },\n                'www': 'https://urdubit.com',\n                'doc': 'https://blinktrade.com/docs',\n            },\n        });\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class vaultoro extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'vaultoro',\n            'name': 'Vaultoro',\n            'countries': 'CH',\n            'rateLimit': 1000,\n            'version': '1',\n            'hasCORS': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766880-f205e870-5ee9-11e7-8fe2-0d5b15880752.jpg',\n                'api': 'https://api.vaultoro.com',\n                'www': 'https://www.vaultoro.com',\n                'doc': 'https://api.vaultoro.com',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'bidandask',\n                        'buyorders',\n                        'latest',\n                        'latesttrades',\n                        'markets',\n                        'orderbook',\n                        'sellorders',\n                        'transactions/day',\n                        'transactions/hour',\n                        'transactions/month',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'balance',\n                        'mytrades',\n                        'orders',\n                    ],\n                    'post': [\n                        'buy/{symbol}/{type}',\n                        'cancel/{id}',\n                        'sell/{symbol}/{type}',\n                        'withdraw',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let result = [];\n        let markets = await this.publicGetMarkets ();\n        let market = markets['data'];\n        let base = market['BaseCurrency'];\n        let quote = market['MarketCurrency'];\n        let symbol = base + '/' + quote;\n        let baseId = base;\n        let quoteId = quote;\n        let id = market['MarketName'];\n        result.push ({\n            'id': id,\n            'symbol': symbol,\n            'base': base,\n            'quote': quote,\n            'baseId': baseId,\n            'quoteId': quoteId,\n            'info': market,\n        });\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetBalance ();\n        let balances = response['data'];\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency_code'];\n            let uppercase = currency.toUpperCase ();\n            let free = balance['cash'];\n            let used = balance['reserved'];\n            let total = this.sum (free, used);\n            let account = {\n                'free': free,\n                'used': used,\n                'total': total,\n            };\n            result[uppercase] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetOrderbook (params);\n        let orderbook = {\n            'bids': response['data'][0]['b'],\n            'asks': response['data'][1]['s'],\n        };\n        let result = this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'Gold_Price', 'Gold_Amount');\n        result['bids'] = this.sortBy (result['bids'], 0, true);\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let quote = await this.publicGetBidandask (params);\n        let bidsLength = quote['bids'].length;\n        let bid = quote['bids'][bidsLength - 1];\n        let ask = quote['asks'][0];\n        let response = await this.publicGetMarkets (params);\n        let ticker = response['data'];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['24hHigh']),\n            'low': parseFloat (ticker['24hLow']),\n            'bid': bid[0],\n            'ask': ask[0],\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['LastPrice']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': parseFloat (ticker['24hVolume']),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = this.parse8601 (trade['Time']);\n        return {\n            'id': undefined,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': undefined,\n            'type': undefined,\n            'side': undefined,\n            'price': trade['Gold_Price'],\n            'amount': trade['Gold_Amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTransactionsDay (params);\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'privatePost' + this.capitalize (side) + 'SymbolType';\n        let response = await this[method] (this.extend ({\n            'symbol': market['quoteId'].toLowerCase (),\n            'type': type,\n            'gld': amount,\n            'price': price || 1,\n        }, params));\n        return {\n            'info': response,\n            'id': response['data']['Order_ID'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelId (this.extend ({\n            'id': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/';\n        if (api == 'public') {\n            url += path;\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            url += this.version + '/' + this.implodeParams (path, params);\n            let query = this.extend ({\n                'nonce': nonce,\n                'apikey': this.apiKey,\n            }, this.omit (params, this.extractParams (path)));\n            url += '?' + this.urlencode (query);\n            headers = {\n                'Content-Type': 'application/json',\n                'X-Signature': this.hmac (this.encode (url), this.encode (this.secret)),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst foxbit = require ('./foxbit.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class vbtc extends foxbit {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'vbtc',\n            'name': 'VBTC',\n            'countries': 'VN',\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27991481-1f53d1d8-6481-11e7-884e-21d17e7939db.jpg',\n                'api': {\n                    'public': 'https://api.blinktrade.com/api',\n                    'private': 'https://api.blinktrade.com/tapi',\n                },\n                'www': 'https://vbtc.exchange',\n                'doc': 'https://blinktrade.com/docs',\n            },\n        });\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class virwox extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'virwox',\n            'name': 'VirWoX',\n            'countries': [ 'AT', 'EU' ],\n            'rateLimit': 1000,\n            'hasCORS': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766894-6da9d360-5eea-11e7-90aa-41f2711b7405.jpg',\n                'api': {\n                    'public': 'http://api.virwox.com/api/json.php',\n                    'private': 'https://www.virwox.com/api/trading.php',\n                },\n                'www': 'https://www.virwox.com',\n                'doc': 'https://www.virwox.com/developers.php',\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': false,\n                'login': true,\n                'password': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'getInstruments',\n                        'getBestPrices',\n                        'getMarketDepth',\n                        'estimateMarketOrder',\n                        'getTradedPriceVolume',\n                        'getRawTradeData',\n                        'getStatistics',\n                        'getTerminalList',\n                        'getGridList',\n                        'getGridStatistics',\n                    ],\n                    'post': [\n                        'getInstruments',\n                        'getBestPrices',\n                        'getMarketDepth',\n                        'estimateMarketOrder',\n                        'getTradedPriceVolume',\n                        'getRawTradeData',\n                        'getStatistics',\n                        'getTerminalList',\n                        'getGridList',\n                        'getGridStatistics',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'cancelOrder',\n                        'getBalances',\n                        'getCommissionDiscount',\n                        'getOrders',\n                        'getTransactions',\n                        'placeOrder',\n                    ],\n                    'post': [\n                        'cancelOrder',\n                        'getBalances',\n                        'getCommissionDiscount',\n                        'getOrders',\n                        'getTransactions',\n                        'placeOrder',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetInstruments ();\n        let keys = Object.keys (markets['result']);\n        let result = [];\n        for (let p = 0; p < keys.length; p++) {\n            let market = markets['result'][keys[p]];\n            let id = market['instrumentID'];\n            let symbol = market['symbol'];\n            let base = market['longCurrency'];\n            let quote = market['shortCurrency'];\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetBalances ();\n        let balances = response['result']['accountList'];\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let total = balance['balance'];\n            let account = {\n                'free': total,\n                'used': 0.0,\n                'total': total,\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchMarketPrice (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicPostGetBestPrices (this.extend ({\n            'symbols': [ symbol ],\n        }, params));\n        let result = response['result'];\n        return {\n            'bid': this.safeFloat (result[0], 'bestBuyPrice'),\n            'ask': this.safeFloat (result[0], 'bestSellPrice'),\n        };\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicPostGetMarketDepth (this.extend ({\n            'symbols': [ symbol ],\n            'buyDepth': 100,\n            'sellDepth': 100,\n        }, params));\n        let orderbook = response['result'][0];\n        return this.parseOrderBook (orderbook, undefined, 'buy', 'sell', 'price', 'volume');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let end = this.milliseconds ();\n        let start = end - 86400000;\n        let response = await this.publicGetTradedPriceVolume (this.extend ({\n            'instrument': symbol,\n            'endDate': this.YmdHMS (end),\n            'startDate': this.YmdHMS (start),\n            'HLOC': 1,\n        }, params));\n        let marketPrice = await this.fetchMarketPrice (symbol, params);\n        let tickers = response['result']['priceVolumeList'];\n        let keys = Object.keys (tickers);\n        let length = keys.length;\n        let lastKey = keys[length - 1];\n        let ticker = tickers[lastKey];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': marketPrice['bid'],\n            'ask': marketPrice['ask'],\n            'vwap': undefined,\n            'open': parseFloat (ticker['open']),\n            'close': parseFloat (ticker['close']),\n            'first': undefined,\n            'last': undefined,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['longVolume']),\n            'quoteVolume': parseFloat (ticker['shortVolume']),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, symbol = undefined) {\n        let sec = this.safeInteger (trade, 'time');\n        let timestamp = sec * 1000;\n        return {\n            'id': trade['tid'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'order': undefined,\n            'symbol': symbol,\n            'type': undefined,\n            'side': undefined,\n            'price': this.safeFloat (trade, 'price'),\n            'amount': this.safeFloat (trade, 'vol'),\n            'fee': undefined,\n            'info': trade,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetRawTradeData (this.extend ({\n            'instrument': symbol,\n            'timespan': 3600,\n        }, params));\n        let result = response['result'];\n        let trades = result['data'];\n        return this.parseTrades (trades, symbol);\n    }\n\n    async createOrder (market, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'instrument': this.symbol (market),\n            'orderType': side.toUpperCase (),\n            'amount': amount,\n        };\n        if (type == 'limit')\n            order['price'] = price;\n        let response = await this.privatePostPlaceOrder (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['orderID'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelOrder (this.extend ({\n            'orderID': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        let auth = {};\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            auth['key'] = this.apiKey;\n            auth['user'] = this.login;\n            auth['pass'] = this.password;\n        }\n        let nonce = this.nonce ();\n        if (method == 'GET') {\n            url += '?' + this.urlencode (this.extend ({\n                'method': path,\n                'id': nonce,\n            }, auth, params));\n        } else {\n            headers = { 'Content-Type': 'application/json' };\n            body = this.json ({\n                'method': path,\n                'params': this.extend (auth, params),\n                'id': nonce,\n            });\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code == 200) {\n            if ((body[0] == '{') || (body[0] == '[')) {\n                let response = JSON.parse (body);\n                if ('result' in response) {\n                    let result = response['result'];\n                    if ('errorCode' in result) {\n                        let errorCode = result['errorCode'];\n                        if (errorCode != 'OK') {\n                            throw new ExchangeError (this.id + ' error returned: ' + body);\n                        }\n                    }\n                } else {\n                    throw new ExchangeError (this.id + ' malformed response: no result in response: ' + body);\n                }\n            } else {\n                // if not a JSON response\n                throw new ExchangeError (this.id + ' returned a non-JSON reply: ' + body);\n            }\n        }\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst liqui = require ('./liqui.js');\nconst { ExchangeError, InsufficientFunds, OrderNotFound, DDoSProtection } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class wex extends liqui {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'wex',\n            'name': 'WEX',\n            'countries': 'NZ', // New Zealand\n            'version': '3',\n            'hasFetchTickers': true,\n            'hasCORS': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30652751-d74ec8f8-9e31-11e7-98c5-71469fcef03e.jpg',\n                'api': {\n                    'public': 'https://wex.nz/api',\n                    'private': 'https://wex.nz/tapi',\n                },\n                'www': 'https://wex.nz',\n                'doc': [\n                    'https://wex.nz/api/3/docs',\n                    'https://wex.nz/tapi/docs',\n                ],\n                'fees': 'https://wex.nz/fees',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'info',\n                        'ticker/{pair}',\n                        'depth/{pair}',\n                        'trades/{pair}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'getInfo',\n                        'Trade',\n                        'ActiveOrders',\n                        'OrderInfo',\n                        'CancelOrder',\n                        'TradeHistory',\n                        'TransHistory',\n                        'CoinDepositAddress',\n                        'WithdrawCoin',\n                        'CreateCoupon',\n                        'RedeemCoupon',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.2 / 100,\n                    'taker': 0.2 / 100,\n                },\n                'funding': {\n                    'withdraw': {\n                        'BTC': 0.001,\n                        'LTC': 0.001,\n                        'NMC': 0.1,\n                        'NVC': 0.1,\n                        'PPC': 0.1,\n                        'DASH': 0.001,\n                        'ETH': 0.003,\n                        'BCH': 0.001,\n                        'ZEC': 0.001,\n                    },\n                },\n            },\n        });\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['updated'] * 1000;\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'sell'),\n            'ask': this.safeFloat (ticker, 'buy'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': this.safeFloat (ticker, 'avg'),\n            'baseVolume': this.safeFloat (ticker, 'vol_cur'),\n            'quoteVolume': this.safeFloat (ticker, 'vol'),\n            'info': ticker,\n        };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code == 200) {\n            if (body[0] != '{') {\n                // response is not JSON\n                throw new ExchangeError (this.id + ' returned a non-JSON reply: ' + body);\n            }\n            let response = JSON.parse (body);\n            if ('success' in response) {\n                if (!response['success']) {\n                    let error = this.safeValue (response, 'error');\n                    if (!error) {\n                        throw new ExchangeError (this.id + ' returned a malformed error: ' + body);\n                    } else if (error == 'bad status') {\n                        throw new OrderNotFound (this.id + ' ' + error);\n                    } else if (error.indexOf ('It is not enough') >= 0) {\n                        throw new InsufficientFunds (this.id + ' ' + error);\n                    } else if (error == 'Requests too often') {\n                        throw new DDoSProtection (this.id + ' ' + error);\n                    } else if (error == 'not available') {\n                        throw new DDoSProtection (this.id + ' ' + error);\n                    } else if (error == 'external service unavailable') {\n                        throw new DDoSProtection (this.id + ' ' + error);\n                    // that's what fetchOpenOrders return if no open orders (fix for #489)\n                    } else if (error != 'no orders') {\n                        throw new ExchangeError (this.id + ' ' + error);\n                    }\n                }\n            }\n        }\n    }\n\n    request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        return this.fetch2 (path, api, method, params, headers, body);\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, NotSupported, AuthenticationError } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class xbtce extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'xbtce',\n            'name': 'xBTCe',\n            'countries': 'RU',\n            'rateLimit': 2000, // responses are cached every 2 seconds\n            'version': 'v1',\n            'hasPublicAPI': false,\n            'hasCORS': false,\n            'hasFetchTickers': true,\n            'hasFetchOHLCV': false,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28059414-e235970c-662c-11e7-8c3a-08e31f78684b.jpg',\n                'api': 'https://cryptottlivewebapi.xbtce.net:8443/api',\n                'www': 'https://www.xbtce.com',\n                'doc': [\n                    'https://www.xbtce.com/tradeapi',\n                    'https://support.xbtce.info/Knowledgebase/Article/View/52/25/xbtce-exchange-api',\n                ],\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'uid': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'currency',\n                        'currency/{filter}',\n                        'level2',\n                        'level2/{filter}',\n                        'quotehistory/{symbol}/{periodicity}/bars/ask',\n                        'quotehistory/{symbol}/{periodicity}/bars/bid',\n                        'quotehistory/{symbol}/level2',\n                        'quotehistory/{symbol}/ticks',\n                        'symbol',\n                        'symbol/{filter}',\n                        'tick',\n                        'tick/{filter}',\n                        'ticker',\n                        'ticker/{filter}',\n                        'tradesession',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'tradeserverinfo',\n                        'tradesession',\n                        'currency',\n                        'currency/{filter}',\n                        'level2',\n                        'level2/{filter}',\n                        'symbol',\n                        'symbol/{filter}',\n                        'tick',\n                        'tick/{filter}',\n                        'account',\n                        'asset',\n                        'asset/{id}',\n                        'position',\n                        'position/{id}',\n                        'trade',\n                        'trade/{id}',\n                        'quotehistory/{symbol}/{periodicity}/bars/ask',\n                        'quotehistory/{symbol}/{periodicity}/bars/ask/info',\n                        'quotehistory/{symbol}/{periodicity}/bars/bid',\n                        'quotehistory/{symbol}/{periodicity}/bars/bid/info',\n                        'quotehistory/{symbol}/level2',\n                        'quotehistory/{symbol}/level2/info',\n                        'quotehistory/{symbol}/periodicities',\n                        'quotehistory/{symbol}/ticks',\n                        'quotehistory/{symbol}/ticks/info',\n                        'quotehistory/cache/{symbol}/{periodicity}/bars/ask',\n                        'quotehistory/cache/{symbol}/{periodicity}/bars/bid',\n                        'quotehistory/cache/{symbol}/level2',\n                        'quotehistory/cache/{symbol}/ticks',\n                        'quotehistory/symbols',\n                        'quotehistory/version',\n                    ],\n                    'post': [\n                        'trade',\n                        'tradehistory',\n                    ],\n                    'put': [\n                        'trade',\n                    ],\n                    'delete': [\n                        'trade',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.privateGetSymbol ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['Symbol'];\n            let base = market['MarginCurrency'];\n            let quote = market['ProfitCurrency'];\n            if (base == 'DSH')\n                base = 'DASH';\n            let symbol = base + '/' + quote;\n            symbol = market['IsTradeAllowed'] ? symbol : id;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privateGetAsset ();\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['Currency'];\n            let uppercase = currency.toUpperCase ();\n            // xbtce names DASH incorrectly as DSH\n            if (uppercase == 'DSH')\n                uppercase = 'DASH';\n            let account = {\n                'free': balance['FreeAmount'],\n                'used': balance['LockedAmount'],\n                'total': balance['Amount'],\n            };\n            result[uppercase] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orderbook = await this.privateGetLevel2Filter (this.extend ({\n            'filter': market['id'],\n        }, params));\n        orderbook = orderbook[0];\n        let timestamp = orderbook['Timestamp'];\n        return this.parseOrderBook (orderbook, timestamp, 'Bids', 'Asks', 'Price', 'Volume');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = 0;\n        let last = undefined;\n        if ('LastBuyTimestamp' in ticker)\n            if (timestamp < ticker['LastBuyTimestamp']) {\n                timestamp = ticker['LastBuyTimestamp'];\n                last = ticker['LastBuyPrice'];\n            }\n        if ('LastSellTimestamp' in ticker)\n            if (timestamp < ticker['LastSellTimestamp']) {\n                timestamp = ticker['LastSellTimestamp'];\n                last = ticker['LastSellPrice'];\n            }\n        if (!timestamp)\n            timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': ticker['DailyBestBuyPrice'],\n            'low': ticker['DailyBestSellPrice'],\n            'bid': ticker['BestBid'],\n            'ask': ticker['BestAsk'],\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': last,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': ticker['DailyTradedTotalVolume'],\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTicker (params);\n        tickers = this.indexBy (tickers, 'Symbol');\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = undefined;\n            let symbol = undefined;\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            } else {\n                let base = id.slice (0, 3);\n                let quote = id.slice (3, 6);\n                if (base == 'DSH')\n                    base = 'DASH';\n                if (quote == 'DSH')\n                    quote = 'DASH';\n                symbol = base + '/' + quote;\n            }\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let tickers = await this.publicGetTickerFilter (this.extend ({\n            'filter': market['id'],\n        }, params));\n        let length = tickers.length;\n        if (length < 1)\n            throw new ExchangeError (this.id + ' fetchTicker returned empty response, xBTCe public API error');\n        tickers = this.indexBy (tickers, 'Symbol');\n        let ticker = tickers[market['id']];\n        return this.parseTicker (ticker, market);\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        // no method for trades?\n        return await this.privateGetTrade (params);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv['Timestamp'],\n            ohlcv['Open'],\n            ohlcv['High'],\n            ohlcv['Low'],\n            ohlcv['Close'],\n            ohlcv['Volume'],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchOHLCV is disabled by the exchange');\n        let minutes = parseInt (timeframe / 60); // 1 minute by default\n        let periodicity = minutes.toString ();\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        if (!since)\n            since = this.seconds () - 86400 * 7; // last day by defulat\n        if (!limit)\n            limit = 1000; // default\n        let response = await this.privateGetQuotehistorySymbolPeriodicityBarsBid (this.extend ({\n            'symbol': market['id'],\n            'periodicity': periodicity,\n            'timestamp': since,\n            'count': limit,\n        }, params));\n        return this.parseOHLCVs (response['Bars'], market, timeframe, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        if (type == 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let response = await this.tapiPostTrade (this.extend ({\n            'pair': this.marketId (symbol),\n            'type': side,\n            'amount': amount,\n            'rate': price,\n        }, params));\n        return {\n            'info': response,\n            'id': response['Id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privateDeleteTrade (this.extend ({\n            'Type': 'Cancel',\n            'Id': id,\n        }, params));\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        if (!this.apiKey)\n            throw new AuthenticationError (this.id + ' requires apiKey for all requests, their public API is always busy');\n        if (!this.uid)\n            throw new AuthenticationError (this.id + ' requires uid property for authentication and trading, their public API is always busy');\n        let url = this.urls['api'] + '/' + this.version;\n        if (api == 'public')\n            url += '/' + api;\n        url += '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            headers = { 'Accept-Encoding': 'gzip, deflate' };\n            let nonce = this.nonce ().toString ();\n            if (method == 'POST') {\n                if (Object.keys (query).length) {\n                    headers['Content-Type'] = 'application/json';\n                    body = this.json (query);\n                } else {\n                    url += '?' + this.urlencode (query);\n                }\n            }\n            let auth = nonce + this.uid + this.apiKey + method + url;\n            if (body)\n                auth += body;\n            let signature = this.hmac (this.encode (auth), this.encode (this.secret), 'sha256', 'base64');\n            let credentials = this.uid + ':' + this.apiKey + ':' + nonce + ':' + this.binaryToString (signature);\n            headers['Authorization'] = 'HMAC ' + credentials;\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","'use strict';\n\n// ---------------------------------------------------------------------------\n\nconst liqui = require ('./liqui.js');\nconst { ExchangeError, InsufficientFunds, DDoSProtection } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class yobit extends liqui {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'yobit',\n            'name': 'YoBit',\n            'countries': 'RU',\n            'rateLimit': 3000, // responses are cached every 2 seconds\n            'version': '3',\n            'hasCORS': false,\n            'hasWithdraw': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766910-cdcbfdae-5eea-11e7-9859-03fea873272d.jpg',\n                'api': {\n                    'public': 'https://yobit.net/api',\n                    'private': 'https://yobit.net/tapi',\n                },\n                'www': 'https://www.yobit.net',\n                'doc': 'https://www.yobit.net/en/api/',\n                'fees': 'https://www.yobit.net/en/fees/',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'depth/{pair}',\n                        'info',\n                        'ticker/{pair}',\n                        'trades/{pair}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'ActiveOrders',\n                        'CancelOrder',\n                        'GetDepositAddress',\n                        'getInfo',\n                        'OrderInfo',\n                        'Trade',\n                        'TradeHistory',\n                        'WithdrawCoinsToAddress',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.002,\n                    'taker': 0.002,\n                },\n                'funding': 0.0,\n                'withdraw': 0.0005,\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        let substitutions = {\n            'AIR': 'AirCoin',\n            'ANI': 'ANICoin',\n            'ANT': 'AntsCoin',\n            'ATM': 'Autumncoin',\n            'BCC': 'BCH',\n            'BTS': 'Bitshares2',\n            'DCT': 'Discount',\n            'DGD': 'DarkGoldCoin',\n            'ICN': 'iCoin',\n            'LIZI': 'LiZi',\n            'LUN': 'LunarCoin',\n            'NAV': 'NavajoCoin',\n            'OMG': 'OMGame',\n            'PAY': 'EPAY',\n            'REP': 'Republicoin',\n        };\n        if (currency in substitutions)\n            return substitutions[currency];\n        return currency;\n    }\n\n    currencyId (commonCode) {\n        let substitutions = {\n            'AirCoin': 'AIR',\n            'ANICoin': 'ANI',\n            'AntsCoin': 'ANT',\n            'Autumncoin': 'ATM',\n            'BCH': 'BCC',\n            'Bitshares2': 'BTS',\n            'Discount': 'DCT',\n            'DarkGoldCoin': 'DGD',\n            'iCoin': 'ICN',\n            'LiZi': 'LIZI',\n            'LunarCoin': 'LUN',\n            'NavajoCoin': 'NAV',\n            'OMGame': 'OMG',\n            'EPAY': 'PAY',\n            'Republicoin': 'REP',\n        };\n        if (commonCode in substitutions)\n            return substitutions[commonCode];\n        return commonCode;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetInfo ();\n        let balances = response['return'];\n        let result = { 'info': balances };\n        let sides = { 'free': 'funds', 'total': 'funds_incl_orders' };\n        let keys = Object.keys (sides);\n        for (let i = 0; i < keys.length; i++) {\n            let key = keys[i];\n            let side = sides[key];\n            if (side in balances) {\n                let currencies = Object.keys (balances[side]);\n                for (let j = 0; j < currencies.length; j++) {\n                    let lowercase = currencies[j];\n                    let uppercase = lowercase.toUpperCase ();\n                    let currency = this.commonCurrencyCode (uppercase);\n                    let account = undefined;\n                    if (currency in result) {\n                        account = result[currency];\n                    } else {\n                        account = this.account ();\n                    }\n                    account[key] = balances[side][lowercase];\n                    if (account['total'] && account['free'])\n                        account['used'] = account['total'] - account['free'];\n                    result[currency] = account;\n                }\n            }\n        }\n        return this.parseBalance (result);\n    }\n\n    async createDepositAddress (currency, params = {}) {\n        let response = await this.fetchDepositAddress (currency, this.extend ({\n            'need_new': 1,\n        }, params));\n        return {\n            'currency': currency,\n            'address': response['address'],\n            'status': 'ok',\n            'info': response['info'],\n        };\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        let currencyId = this.currencyId (currency);\n        let request = {\n            'coinName': currencyId,\n            'need_new': 0,\n        };\n        let response = await this.privatePostGetDepositAddress (this.extend (request, params));\n        let address = this.safeString (response['return'], 'address');\n        return {\n            'currency': currency,\n            'address': address,\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostWithdrawCoinsToAddress (this.extend ({\n            'coinName': currency,\n            'amount': amount,\n            'address': address,\n        }, params));\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('success' in response) {\n            if (!response['success']) {\n                if (response['error'].indexOf ('Insufficient funds') >= 0) { // not enougTh is a typo inside Liqui's own API...\n                    throw new InsufficientFunds (this.id + ' ' + this.json (response));\n                } else if (response['error'] === 'Requests too often') {\n                    throw new DDoSProtection (this.id + ' ' + this.json (response));\n                } else if ((response['error'] === 'not available') || (response['error'] === 'external service unavailable')) {\n                    throw new DDoSProtection (this.id + ' ' + this.json (response));\n                } else {\n                    throw new ExchangeError (this.id + ' ' + this.json (response));\n                }\n            }\n        }\n        return response;\n    }\n\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst acx = require ('./acx.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class yunbi extends acx {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'yunbi',\n            'name': 'YUNBI',\n            'countries': 'CN',\n            'rateLimit': 1000,\n            'version': 'v2',\n            'hasCORS': false,\n            'hasFetchTickers': true,\n            'hasFetchOHLCV': true,\n            'timeframes': {\n                '1m': '1',\n                '5m': '5',\n                '15m': '15',\n                '30m': '30',\n                '1h': '60',\n                '2h': '120',\n                '4h': '240',\n                '12h': '720',\n                '1d': '1440',\n                '3d': '4320',\n                '1w': '10080',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28570548-4d646c40-7147-11e7-9cf6-839b93e6d622.jpg',\n                'extension': '.json', // default extension appended to endpoint URLs\n                'api': 'https://yunbi.com',\n                'www': 'https://yunbi.com',\n                'doc': [\n                    'https://yunbi.com/documents/api/guide',\n                    'https://yunbi.com/swagger/',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'tickers',\n                        'tickers/{market}',\n                        'markets',\n                        'order_book',\n                        'k',\n                        'depth',\n                        'trades',\n                        'k_with_pending_trades',\n                        'timestamp',\n                        'addresses/{address}',\n                        'partners/orders/{id}/trades',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'deposits',\n                        'members/me',\n                        'deposit',\n                        'deposit_address',\n                        'order',\n                        'orders',\n                        'trades/my',\n                    ],\n                    'post': [\n                        'order/delete',\n                        'orders',\n                        'orders/multi',\n                        'orders/clear',\n                    ],\n                },\n            },\n        });\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class zaif extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'zaif',\n            'name': 'Zaif',\n            'countries': 'JP',\n            'rateLimit': 2000,\n            'version': '1',\n            'hasCORS': false,\n            'hasFetchOpenOrders': true,\n            'hasFetchClosedOrders': true,\n            'hasWithdraw': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766927-39ca2ada-5eeb-11e7-972f-1b4199518ca6.jpg',\n                'api': 'https://api.zaif.jp',\n                'www': 'https://zaif.jp',\n                'doc': [\n                    'http://techbureau-api-document.readthedocs.io/ja/latest/index.html',\n                    'https://corp.zaif.jp/api-docs',\n                    'https://corp.zaif.jp/api-docs/api_links',\n                    'https://www.npmjs.com/package/zaif.jp',\n                    'https://github.com/you21979/node-zaif',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'depth/{pair}',\n                        'currencies/{pair}',\n                        'currencies/all',\n                        'currency_pairs/{pair}',\n                        'currency_pairs/all',\n                        'last_price/{pair}',\n                        'ticker/{pair}',\n                        'trades/{pair}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'active_orders',\n                        'cancel_order',\n                        'deposit_history',\n                        'get_id_info',\n                        'get_info',\n                        'get_info2',\n                        'get_personal_info',\n                        'trade',\n                        'trade_history',\n                        'withdraw',\n                        'withdraw_history',\n                    ],\n                },\n                'ecapi': {\n                    'post': [\n                        'createInvoice',\n                        'getInvoice',\n                        'getInvoiceIdsByOrderNumber',\n                        'cancelInvoice',\n                    ],\n                },\n                'tlapi': {\n                    'post': [\n                        'get_positions',\n                        'position_history',\n                        'active_positions',\n                        'create_position',\n                        'change_position',\n                        'cancel_position',\n                    ],\n                },\n                'fapi': {\n                    'get': [\n                        'groups/{group_id}',\n                        'last_price/{group_id}/{pair}',\n                        'ticker/{group_id}/{pair}',\n                        'trades/{group_id}/{pair}',\n                        'depth/{group_id}/{pair}',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetCurrencyPairsAll ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['currency_pair'];\n            let symbol = market['name'];\n            let [ base, quote ] = symbol.split ('/');\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetInfo ();\n        let balances = response['return'];\n        let result = { 'info': balances };\n        let currencies = Object.keys (balances['funds']);\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let balance = balances['funds'][currency];\n            let uppercase = currency.toUpperCase ();\n            let account = {\n                'free': balance,\n                'used': 0.0,\n                'total': balance,\n            };\n            if ('deposit' in balances) {\n                if (currency in balances['deposit']) {\n                    account['total'] = balances['deposit'][currency];\n                    account['used'] = account['total'] - account['free'];\n                }\n            }\n            result[uppercase] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetDepthPair (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let ticker = await this.publicGetTickerPair (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        let timestamp = this.milliseconds ();\n        let vwap = ticker['vwap'];\n        let baseVolume = ticker['volume'];\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': ticker['high'],\n            'low': ticker['low'],\n            'bid': ticker['bid'],\n            'ask': ticker['ask'],\n            'vwap': vwap,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': ticker['last'],\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let side = (trade['trade_type'] === 'bid') ? 'buy' : 'sell';\n        let timestamp = trade['date'] * 1000;\n        let id = this.safeString (trade, 'id');\n        id = this.safeString (trade, 'tid', id);\n        if (!market)\n            market = this.markets_by_id[trade['currency_pair']];\n        return {\n            'id': id.toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': side,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTradesPair (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        if (type === 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let response = await this.privatePostTrade (this.extend ({\n            'currency_pair': this.marketId (symbol),\n            'action': (side === 'buy') ? 'bid' : 'ask',\n            'amount': amount,\n            'price': price,\n        }, params));\n        return {\n            'info': response,\n            'id': response['return']['order_id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelOrder (this.extend ({\n            'order_id': id,\n        }, params));\n    }\n\n    parseOrder (order, market = undefined) {\n        let side = (order['action'] === 'bid') ? 'buy' : 'sell';\n        let timestamp = parseInt (order['timestamp']) * 1000;\n        if (!market)\n            market = this.markets_by_id[order['currency_pair']];\n        let price = order['price'];\n        let amount = order['amount'];\n        return {\n            'id': order['id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': 'open',\n            'symbol': market['symbol'],\n            'type': 'limit',\n            'side': side,\n            'price': price,\n            'cost': price * amount,\n            'amount': amount,\n            'filled': undefined,\n            'remaining': undefined,\n            'trades': undefined,\n            'fee': undefined,\n        };\n    }\n\n    parseOrders (orders, market = undefined, since = undefined, limit = undefined) {\n        let ids = Object.keys (orders);\n        let result = [];\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let order = orders[id];\n            let extended = this.extend (order, { 'id': id });\n            result.push (this.parseOrder (extended, market));\n        }\n        return this.filterBySinceLimit (result, since, limit);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        let request = {\n            // 'is_token': false,\n            // 'is_token_both': false,\n        };\n        if (symbol) {\n            market = this.market (symbol);\n            request['currency_pair'] = market['id'];\n        }\n        let response = await this.privatePostActiveOrders (this.extend (request, params));\n        return this.parseOrders (response['return'], market, since, limit);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        let request = {\n            // 'from': 0,\n            // 'count': 1000,\n            // 'from_id': 0,\n            // 'end_id': 1000,\n            // 'order': 'DESC',\n            // 'since': 1503821051,\n            // 'end': 1503821051,\n            // 'is_token': false,\n        };\n        if (symbol) {\n            market = this.market (symbol);\n            request['currency_pair'] = market['id'];\n        }\n        let response = await this.privatePostTradeHistory (this.extend (request, params));\n        return this.parseOrders (response['return'], market, since, limit);\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        if (currency === 'JPY')\n            throw new ExchangeError (this.id + ' does not allow ' + currency + ' withdrawals');\n        let result = await this.privatePostWithdraw (this.extend ({\n            'currency': currency,\n            'amount': amount,\n            'address': address,\n            // 'message': 'Hi!', // XEM only\n            // 'opt_fee': 0.003, // BTC and MONA only\n        }, params));\n        return {\n            'info': result,\n            'id': result['return']['txid'],\n            'fee': result['return']['fee'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/';\n        if (api === 'public') {\n            url += 'api/' + this.version + '/' + this.implodeParams (path, params);\n        } else if (api === 'fapi') {\n            url += 'fapi/' + this.version + '/' + this.implodeParams (path, params);\n        } else {\n            this.checkRequiredCredentials ();\n            if (api === 'ecapi') {\n                url += 'ecapi';\n            } else if (api === 'tlapi') {\n                url += 'tlapi';\n            } else {\n                url += 'tapi';\n            }\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({\n                'method': path,\n                'nonce': nonce,\n            }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Key': this.apiKey,\n                'Sign': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'api', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response)\n            throw new ExchangeError (this.id + ' ' + response['error']);\n        if ('success' in response)\n            if (!response['success'])\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class zb extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'zb',\n            'name': 'ZB',\n            'countries': 'CN',\n            'rateLimit': 1000,\n            'version': 'v1',\n            'hasCORS': false,\n            'hasFetchOrder': true,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/32859187-cd5214f0-ca5e-11e7-967d-96568e2e2bd1.jpg',\n                'api': {\n                    'public': 'http://api.zb.com/data', // no https for public API\n                    'private': 'https://trade.zb.com/api',\n                },\n                'www': 'https://trade.zb.com/api',\n                'doc': 'https://www.zb.com/i/developer',\n                'fees': 'https://www.zb.com/i/rate',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'markets',\n                        'ticker',\n                        'depth',\n                        'trades',\n                        'kline',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'order',\n                        'cancelOrder',\n                        'getOrder',\n                        'getOrders',\n                        'getOrdersNew',\n                        'getOrdersIgnoreTradeType',\n                        'getUnfinishedOrdersIgnoreTradeType',\n                        'getAccountInfo',\n                        'getUserAddress',\n                        'getWithdrawAddress',\n                        'getWithdrawRecord',\n                        'getChargeRecord',\n                        'getCnyWithdrawRecord',\n                        'getCnyChargeRecord',\n                        'withdraw',\n                    ],\n                },\n            },\n            'fees': {\n                'funding': {\n                    'withdraw': {\n                        'BTC': 0.0001,\n                        'BCH': 0.0006,\n                        'LTC': 0.005,\n                        'ETH': 0.01,\n                        'ETC': 0.01,\n                        'BTS': 3,\n                        'EOS': 1,\n                        'QTUM': 0.01,\n                        'HSR': 0.001,\n                        'XRP': 0.1,\n                        'USDT': '0.1%',\n                        'QCASH': 5,\n                        'DASH': 0.002,\n                        'BCD': 0,\n                        'UBTC': 0,\n                        'SBTC': 0,\n                        'INK': 20,\n                        'TV': 0.1,\n                        'BTH': 0,\n                        'BCX': 0,\n                        'LBTC': 0,\n                        'CHAT': 20,\n                        'bitCNY': 20,\n                        'HLC': 20,\n                        'BTP': 0,\n                        'BCW': 0,\n                    },\n                },\n                'trading': {\n                },\n            },\n        });\n    }\n\n    getTradingFeeFromBaseQuote (base, quote) {\n        // base: quote\n        let fees = {\n            'BTC': { 'USDT': 0.0 },\n            'BCH': { 'BTC': 0.001, 'USDT': 0.001 },\n            'LTC': { 'BTC': 0.001, 'USDT': 0.0 },\n            'ETH': { 'BTC': 0.001, 'USDT': 0.0 },\n            'ETC': { 'BTC': 0.001, 'USDT': 0.0 },\n            'BTS': { 'BTC': 0.001, 'USDT': 0.001 },\n            'EOS': { 'BTC': 0.001, 'USDT': 0.001 },\n            'HSR': { 'BTC': 0.001, 'USDT': 0.001 },\n            'QTUM': { 'BTC': 0.001, 'USDT': 0.001 },\n            'USDT': { 'BTC': 0.0 },\n        };\n        if (base in fees) {\n            let quoteFees = fees[base];\n            if (quote in quoteFees)\n                return quoteFees[quote];\n        }\n        return undefined;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetMarkets ();\n        let keys = Object.keys (markets);\n        let result = [];\n        for (let i = 0; i < keys.length; i++) {\n            let id = keys[i];\n            let market = markets[id];\n            let [ baseId, quoteId ] = id.split ('_');\n            let base = this.commonCurrencyCode (baseId.toUpperCase ());\n            let quote = this.commonCurrencyCode (quoteId.toUpperCase ());\n            let symbol = base + '/' + quote;\n            let fee = this.getTradingFeeFromBaseQuote (base, quote);\n            let precision = {\n                'amount': market['amountScale'],\n                'price': market['priceScale'],\n            };\n            let lot = Math.pow (10, -precision['amount']);\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'baseId': baseId,\n                'quoteId': quoteId,\n                'base': base,\n                'quote': quote,\n                'info': market,\n                'maker': fee,\n                'taker': fee,\n                'lot': lot,\n                'active': true,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision['price']),\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': 0,\n                        'max': undefined,\n                    },\n                },\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetAccountInfo ();\n        let balances = response['result'];\n        let result = { 'info': balances };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            if (currency in balances['balance'])\n                account['free'] = parseFloat (balances['balance'][currency]['amount']);\n            if (currency in balances['frozen'])\n                account['used'] = parseFloat (balances['frozen'][currency]['amount']);\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    getMarketFieldName () {\n        return 'market';\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let marketFieldName = this.getMarketFieldName ();\n        let request = {};\n        request[marketFieldName] = market['id'];\n        let orderbook = await this.publicGetDepth (this.extend (request, params));\n        let timestamp = this.milliseconds ();\n        let bids = undefined;\n        let asks = undefined;\n        if ('bids' in orderbook)\n            bids = orderbook['bids'];\n        if ('asks' in orderbook)\n            asks = orderbook['asks'];\n        let result = {\n            'bids': bids,\n            'asks': asks,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n        };\n        if (result['bids'])\n            result['bids'] = this.sortBy (result['bids'], 0, true);\n        if (result['asks'])\n            result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let marketFieldName = this.getMarketFieldName ();\n        let request = {};\n        request[marketFieldName] = market['id'];\n        let response = await this.publicGetTicker (this.extend (request, params));\n        let ticker = response['ticker'];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy']),\n            'ask': parseFloat (ticker['sell']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['vol']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = trade['date'] * 1000;\n        let side = (trade['trade_type'] == 'bid') ? 'buy' : 'sell';\n        return {\n            'info': trade,\n            'id': trade['tid'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': side,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let marketFieldName = this.getMarketFieldName ();\n        let request = {};\n        request[marketFieldName] = market['id'];\n        let response = await this.publicGetTrades (this.extend (request, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let paramString = '&price=' + price.toString ();\n        paramString += '&amount=' + amount.toString ();\n        let tradeType = (side == 'buy') ? '1' : '0';\n        paramString += '&tradeType=' + tradeType;\n        paramString += '&currency=' + this.marketId (symbol);\n        let response = await this.privatePostOrder (paramString);\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let paramString = '&id=' + id.toString ();\n        if ('currency' in params)\n            paramString += '&currency=' + params['currency'];\n        return await this.privatePostCancelOrder (paramString);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let paramString = '&id=' + id.toString ();\n        if ('currency' in params)\n            paramString += '&currency=' + params['currency'];\n        return await this.privatePostGetOrder (paramString);\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        if (api == 'public') {\n            url += '/' + this.version + '/' + path;\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let auth = 'accesskey=' + this.apiKey;\n            auth += '&method=' + path;\n            let secret = this.hash (this.encode (this.secret), 'sha1');\n            let signature = this.hmac (this.encode (auth), this.encode (secret), 'md5');\n            let suffix = 'sign=' + signature + '&reqTime=' + nonce.toString ();\n            url += '/' + path + '?' + auth + '&' + suffix;\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (api == 'private')\n            if ('code' in response)\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./enc-base64\"), require(\"./md5\"), require(\"./evpkdf\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./enc-base64\", \"./md5\", \"./evpkdf\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var BlockCipher = C_lib.BlockCipher;\n\t    var C_algo = C.algo;\n\n\t    // Lookup tables\n\t    var SBOX = [];\n\t    var INV_SBOX = [];\n\t    var SUB_MIX_0 = [];\n\t    var SUB_MIX_1 = [];\n\t    var SUB_MIX_2 = [];\n\t    var SUB_MIX_3 = [];\n\t    var INV_SUB_MIX_0 = [];\n\t    var INV_SUB_MIX_1 = [];\n\t    var INV_SUB_MIX_2 = [];\n\t    var INV_SUB_MIX_3 = [];\n\n\t    // Compute lookup tables\n\t    (function () {\n\t        // Compute double table\n\t        var d = [];\n\t        for (var i = 0; i < 256; i++) {\n\t            if (i < 128) {\n\t                d[i] = i << 1;\n\t            } else {\n\t                d[i] = (i << 1) ^ 0x11b;\n\t            }\n\t        }\n\n\t        // Walk GF(2^8)\n\t        var x = 0;\n\t        var xi = 0;\n\t        for (var i = 0; i < 256; i++) {\n\t            // Compute sbox\n\t            var sx = xi ^ (xi << 1) ^ (xi << 2) ^ (xi << 3) ^ (xi << 4);\n\t            sx = (sx >>> 8) ^ (sx & 0xff) ^ 0x63;\n\t            SBOX[x] = sx;\n\t            INV_SBOX[sx] = x;\n\n\t            // Compute multiplication\n\t            var x2 = d[x];\n\t            var x4 = d[x2];\n\t            var x8 = d[x4];\n\n\t            // Compute sub bytes, mix columns tables\n\t            var t = (d[sx] * 0x101) ^ (sx * 0x1010100);\n\t            SUB_MIX_0[x] = (t << 24) | (t >>> 8);\n\t            SUB_MIX_1[x] = (t << 16) | (t >>> 16);\n\t            SUB_MIX_2[x] = (t << 8)  | (t >>> 24);\n\t            SUB_MIX_3[x] = t;\n\n\t            // Compute inv sub bytes, inv mix columns tables\n\t            var t = (x8 * 0x1010101) ^ (x4 * 0x10001) ^ (x2 * 0x101) ^ (x * 0x1010100);\n\t            INV_SUB_MIX_0[sx] = (t << 24) | (t >>> 8);\n\t            INV_SUB_MIX_1[sx] = (t << 16) | (t >>> 16);\n\t            INV_SUB_MIX_2[sx] = (t << 8)  | (t >>> 24);\n\t            INV_SUB_MIX_3[sx] = t;\n\n\t            // Compute next counter\n\t            if (!x) {\n\t                x = xi = 1;\n\t            } else {\n\t                x = x2 ^ d[d[d[x8 ^ x2]]];\n\t                xi ^= d[d[xi]];\n\t            }\n\t        }\n\t    }());\n\n\t    // Precomputed Rcon lookup\n\t    var RCON = [0x00, 0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 0x1b, 0x36];\n\n\t    /**\n\t     * AES block cipher algorithm.\n\t     */\n\t    var AES = C_algo.AES = BlockCipher.extend({\n\t        _doReset: function () {\n\t            // Skip reset of nRounds has been set before and key did not change\n\t            if (this._nRounds && this._keyPriorReset === this._key) {\n\t                return;\n\t            }\n\n\t            // Shortcuts\n\t            var key = this._keyPriorReset = this._key;\n\t            var keyWords = key.words;\n\t            var keySize = key.sigBytes / 4;\n\n\t            // Compute number of rounds\n\t            var nRounds = this._nRounds = keySize + 6;\n\n\t            // Compute number of key schedule rows\n\t            var ksRows = (nRounds + 1) * 4;\n\n\t            // Compute key schedule\n\t            var keySchedule = this._keySchedule = [];\n\t            for (var ksRow = 0; ksRow < ksRows; ksRow++) {\n\t                if (ksRow < keySize) {\n\t                    keySchedule[ksRow] = keyWords[ksRow];\n\t                } else {\n\t                    var t = keySchedule[ksRow - 1];\n\n\t                    if (!(ksRow % keySize)) {\n\t                        // Rot word\n\t                        t = (t << 8) | (t >>> 24);\n\n\t                        // Sub word\n\t                        t = (SBOX[t >>> 24] << 24) | (SBOX[(t >>> 16) & 0xff] << 16) | (SBOX[(t >>> 8) & 0xff] << 8) | SBOX[t & 0xff];\n\n\t                        // Mix Rcon\n\t                        t ^= RCON[(ksRow / keySize) | 0] << 24;\n\t                    } else if (keySize > 6 && ksRow % keySize == 4) {\n\t                        // Sub word\n\t                        t = (SBOX[t >>> 24] << 24) | (SBOX[(t >>> 16) & 0xff] << 16) | (SBOX[(t >>> 8) & 0xff] << 8) | SBOX[t & 0xff];\n\t                    }\n\n\t                    keySchedule[ksRow] = keySchedule[ksRow - keySize] ^ t;\n\t                }\n\t            }\n\n\t            // Compute inv key schedule\n\t            var invKeySchedule = this._invKeySchedule = [];\n\t            for (var invKsRow = 0; invKsRow < ksRows; invKsRow++) {\n\t                var ksRow = ksRows - invKsRow;\n\n\t                if (invKsRow % 4) {\n\t                    var t = keySchedule[ksRow];\n\t                } else {\n\t                    var t = keySchedule[ksRow - 4];\n\t                }\n\n\t                if (invKsRow < 4 || ksRow <= 4) {\n\t                    invKeySchedule[invKsRow] = t;\n\t                } else {\n\t                    invKeySchedule[invKsRow] = INV_SUB_MIX_0[SBOX[t >>> 24]] ^ INV_SUB_MIX_1[SBOX[(t >>> 16) & 0xff]] ^\n\t                                               INV_SUB_MIX_2[SBOX[(t >>> 8) & 0xff]] ^ INV_SUB_MIX_3[SBOX[t & 0xff]];\n\t                }\n\t            }\n\t        },\n\n\t        encryptBlock: function (M, offset) {\n\t            this._doCryptBlock(M, offset, this._keySchedule, SUB_MIX_0, SUB_MIX_1, SUB_MIX_2, SUB_MIX_3, SBOX);\n\t        },\n\n\t        decryptBlock: function (M, offset) {\n\t            // Swap 2nd and 4th rows\n\t            var t = M[offset + 1];\n\t            M[offset + 1] = M[offset + 3];\n\t            M[offset + 3] = t;\n\n\t            this._doCryptBlock(M, offset, this._invKeySchedule, INV_SUB_MIX_0, INV_SUB_MIX_1, INV_SUB_MIX_2, INV_SUB_MIX_3, INV_SBOX);\n\n\t            // Inv swap 2nd and 4th rows\n\t            var t = M[offset + 1];\n\t            M[offset + 1] = M[offset + 3];\n\t            M[offset + 3] = t;\n\t        },\n\n\t        _doCryptBlock: function (M, offset, keySchedule, SUB_MIX_0, SUB_MIX_1, SUB_MIX_2, SUB_MIX_3, SBOX) {\n\t            // Shortcut\n\t            var nRounds = this._nRounds;\n\n\t            // Get input, add round key\n\t            var s0 = M[offset]     ^ keySchedule[0];\n\t            var s1 = M[offset + 1] ^ keySchedule[1];\n\t            var s2 = M[offset + 2] ^ keySchedule[2];\n\t            var s3 = M[offset + 3] ^ keySchedule[3];\n\n\t            // Key schedule row counter\n\t            var ksRow = 4;\n\n\t            // Rounds\n\t            for (var round = 1; round < nRounds; round++) {\n\t                // Shift rows, sub bytes, mix columns, add round key\n\t                var t0 = SUB_MIX_0[s0 >>> 24] ^ SUB_MIX_1[(s1 >>> 16) & 0xff] ^ SUB_MIX_2[(s2 >>> 8) & 0xff] ^ SUB_MIX_3[s3 & 0xff] ^ keySchedule[ksRow++];\n\t                var t1 = SUB_MIX_0[s1 >>> 24] ^ SUB_MIX_1[(s2 >>> 16) & 0xff] ^ SUB_MIX_2[(s3 >>> 8) & 0xff] ^ SUB_MIX_3[s0 & 0xff] ^ keySchedule[ksRow++];\n\t                var t2 = SUB_MIX_0[s2 >>> 24] ^ SUB_MIX_1[(s3 >>> 16) & 0xff] ^ SUB_MIX_2[(s0 >>> 8) & 0xff] ^ SUB_MIX_3[s1 & 0xff] ^ keySchedule[ksRow++];\n\t                var t3 = SUB_MIX_0[s3 >>> 24] ^ SUB_MIX_1[(s0 >>> 16) & 0xff] ^ SUB_MIX_2[(s1 >>> 8) & 0xff] ^ SUB_MIX_3[s2 & 0xff] ^ keySchedule[ksRow++];\n\n\t                // Update state\n\t                s0 = t0;\n\t                s1 = t1;\n\t                s2 = t2;\n\t                s3 = t3;\n\t            }\n\n\t            // Shift rows, sub bytes, add round key\n\t            var t0 = ((SBOX[s0 >>> 24] << 24) | (SBOX[(s1 >>> 16) & 0xff] << 16) | (SBOX[(s2 >>> 8) & 0xff] << 8) | SBOX[s3 & 0xff]) ^ keySchedule[ksRow++];\n\t            var t1 = ((SBOX[s1 >>> 24] << 24) | (SBOX[(s2 >>> 16) & 0xff] << 16) | (SBOX[(s3 >>> 8) & 0xff] << 8) | SBOX[s0 & 0xff]) ^ keySchedule[ksRow++];\n\t            var t2 = ((SBOX[s2 >>> 24] << 24) | (SBOX[(s3 >>> 16) & 0xff] << 16) | (SBOX[(s0 >>> 8) & 0xff] << 8) | SBOX[s1 & 0xff]) ^ keySchedule[ksRow++];\n\t            var t3 = ((SBOX[s3 >>> 24] << 24) | (SBOX[(s0 >>> 16) & 0xff] << 16) | (SBOX[(s1 >>> 8) & 0xff] << 8) | SBOX[s2 & 0xff]) ^ keySchedule[ksRow++];\n\n\t            // Set output\n\t            M[offset]     = t0;\n\t            M[offset + 1] = t1;\n\t            M[offset + 2] = t2;\n\t            M[offset + 3] = t3;\n\t        },\n\n\t        keySize: 256/32\n\t    });\n\n\t    /**\n\t     * Shortcut functions to the cipher's object interface.\n\t     *\n\t     * @example\n\t     *\n\t     *     var ciphertext = CryptoJS.AES.encrypt(message, key, cfg);\n\t     *     var plaintext  = CryptoJS.AES.decrypt(ciphertext, key, cfg);\n\t     */\n\t    C.AES = BlockCipher._createHelper(AES);\n\t}());\n\n\n\treturn CryptoJS.AES;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./evpkdf\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./evpkdf\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * Cipher core components.\n\t */\n\tCryptoJS.lib.Cipher || (function (undefined) {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var Base = C_lib.Base;\n\t    var WordArray = C_lib.WordArray;\n\t    var BufferedBlockAlgorithm = C_lib.BufferedBlockAlgorithm;\n\t    var C_enc = C.enc;\n\t    var Utf8 = C_enc.Utf8;\n\t    var Base64 = C_enc.Base64;\n\t    var C_algo = C.algo;\n\t    var EvpKDF = C_algo.EvpKDF;\n\n\t    /**\n\t     * Abstract base cipher template.\n\t     *\n\t     * @property {number} keySize This cipher's key size. Default: 4 (128 bits)\n\t     * @property {number} ivSize This cipher's IV size. Default: 4 (128 bits)\n\t     * @property {number} _ENC_XFORM_MODE A constant representing encryption mode.\n\t     * @property {number} _DEC_XFORM_MODE A constant representing decryption mode.\n\t     */\n\t    var Cipher = C_lib.Cipher = BufferedBlockAlgorithm.extend({\n\t        /**\n\t         * Configuration options.\n\t         *\n\t         * @property {WordArray} iv The IV to use for this operation.\n\t         */\n\t        cfg: Base.extend(),\n\n\t        /**\n\t         * Creates this cipher in encryption mode.\n\t         *\n\t         * @param {WordArray} key The key.\n\t         * @param {Object} cfg (Optional) The configuration options to use for this operation.\n\t         *\n\t         * @return {Cipher} A cipher instance.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var cipher = CryptoJS.algo.AES.createEncryptor(keyWordArray, { iv: ivWordArray });\n\t         */\n\t        createEncryptor: function (key, cfg) {\n\t            return this.create(this._ENC_XFORM_MODE, key, cfg);\n\t        },\n\n\t        /**\n\t         * Creates this cipher in decryption mode.\n\t         *\n\t         * @param {WordArray} key The key.\n\t         * @param {Object} cfg (Optional) The configuration options to use for this operation.\n\t         *\n\t         * @return {Cipher} A cipher instance.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var cipher = CryptoJS.algo.AES.createDecryptor(keyWordArray, { iv: ivWordArray });\n\t         */\n\t        createDecryptor: function (key, cfg) {\n\t            return this.create(this._DEC_XFORM_MODE, key, cfg);\n\t        },\n\n\t        /**\n\t         * Initializes a newly created cipher.\n\t         *\n\t         * @param {number} xformMode Either the encryption or decryption transormation mode constant.\n\t         * @param {WordArray} key The key.\n\t         * @param {Object} cfg (Optional) The configuration options to use for this operation.\n\t         *\n\t         * @example\n\t         *\n\t         *     var cipher = CryptoJS.algo.AES.create(CryptoJS.algo.AES._ENC_XFORM_MODE, keyWordArray, { iv: ivWordArray });\n\t         */\n\t        init: function (xformMode, key, cfg) {\n\t            // Apply config defaults\n\t            this.cfg = this.cfg.extend(cfg);\n\n\t            // Store transform mode and key\n\t            this._xformMode = xformMode;\n\t            this._key = key;\n\n\t            // Set initial values\n\t            this.reset();\n\t        },\n\n\t        /**\n\t         * Resets this cipher to its initial state.\n\t         *\n\t         * @example\n\t         *\n\t         *     cipher.reset();\n\t         */\n\t        reset: function () {\n\t            // Reset data buffer\n\t            BufferedBlockAlgorithm.reset.call(this);\n\n\t            // Perform concrete-cipher logic\n\t            this._doReset();\n\t        },\n\n\t        /**\n\t         * Adds data to be encrypted or decrypted.\n\t         *\n\t         * @param {WordArray|string} dataUpdate The data to encrypt or decrypt.\n\t         *\n\t         * @return {WordArray} The data after processing.\n\t         *\n\t         * @example\n\t         *\n\t         *     var encrypted = cipher.process('data');\n\t         *     var encrypted = cipher.process(wordArray);\n\t         */\n\t        process: function (dataUpdate) {\n\t            // Append\n\t            this._append(dataUpdate);\n\n\t            // Process available blocks\n\t            return this._process();\n\t        },\n\n\t        /**\n\t         * Finalizes the encryption or decryption process.\n\t         * Note that the finalize operation is effectively a destructive, read-once operation.\n\t         *\n\t         * @param {WordArray|string} dataUpdate The final data to encrypt or decrypt.\n\t         *\n\t         * @return {WordArray} The data after final processing.\n\t         *\n\t         * @example\n\t         *\n\t         *     var encrypted = cipher.finalize();\n\t         *     var encrypted = cipher.finalize('data');\n\t         *     var encrypted = cipher.finalize(wordArray);\n\t         */\n\t        finalize: function (dataUpdate) {\n\t            // Final data update\n\t            if (dataUpdate) {\n\t                this._append(dataUpdate);\n\t            }\n\n\t            // Perform concrete-cipher logic\n\t            var finalProcessedData = this._doFinalize();\n\n\t            return finalProcessedData;\n\t        },\n\n\t        keySize: 128/32,\n\n\t        ivSize: 128/32,\n\n\t        _ENC_XFORM_MODE: 1,\n\n\t        _DEC_XFORM_MODE: 2,\n\n\t        /**\n\t         * Creates shortcut functions to a cipher's object interface.\n\t         *\n\t         * @param {Cipher} cipher The cipher to create a helper for.\n\t         *\n\t         * @return {Object} An object with encrypt and decrypt shortcut functions.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var AES = CryptoJS.lib.Cipher._createHelper(CryptoJS.algo.AES);\n\t         */\n\t        _createHelper: (function () {\n\t            function selectCipherStrategy(key) {\n\t                if (typeof key == 'string') {\n\t                    return PasswordBasedCipher;\n\t                } else {\n\t                    return SerializableCipher;\n\t                }\n\t            }\n\n\t            return function (cipher) {\n\t                return {\n\t                    encrypt: function (message, key, cfg) {\n\t                        return selectCipherStrategy(key).encrypt(cipher, message, key, cfg);\n\t                    },\n\n\t                    decrypt: function (ciphertext, key, cfg) {\n\t                        return selectCipherStrategy(key).decrypt(cipher, ciphertext, key, cfg);\n\t                    }\n\t                };\n\t            };\n\t        }())\n\t    });\n\n\t    /**\n\t     * Abstract base stream cipher template.\n\t     *\n\t     * @property {number} blockSize The number of 32-bit words this cipher operates on. Default: 1 (32 bits)\n\t     */\n\t    var StreamCipher = C_lib.StreamCipher = Cipher.extend({\n\t        _doFinalize: function () {\n\t            // Process partial blocks\n\t            var finalProcessedBlocks = this._process(!!'flush');\n\n\t            return finalProcessedBlocks;\n\t        },\n\n\t        blockSize: 1\n\t    });\n\n\t    /**\n\t     * Mode namespace.\n\t     */\n\t    var C_mode = C.mode = {};\n\n\t    /**\n\t     * Abstract base block cipher mode template.\n\t     */\n\t    var BlockCipherMode = C_lib.BlockCipherMode = Base.extend({\n\t        /**\n\t         * Creates this mode for encryption.\n\t         *\n\t         * @param {Cipher} cipher A block cipher instance.\n\t         * @param {Array} iv The IV words.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var mode = CryptoJS.mode.CBC.createEncryptor(cipher, iv.words);\n\t         */\n\t        createEncryptor: function (cipher, iv) {\n\t            return this.Encryptor.create(cipher, iv);\n\t        },\n\n\t        /**\n\t         * Creates this mode for decryption.\n\t         *\n\t         * @param {Cipher} cipher A block cipher instance.\n\t         * @param {Array} iv The IV words.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var mode = CryptoJS.mode.CBC.createDecryptor(cipher, iv.words);\n\t         */\n\t        createDecryptor: function (cipher, iv) {\n\t            return this.Decryptor.create(cipher, iv);\n\t        },\n\n\t        /**\n\t         * Initializes a newly created mode.\n\t         *\n\t         * @param {Cipher} cipher A block cipher instance.\n\t         * @param {Array} iv The IV words.\n\t         *\n\t         * @example\n\t         *\n\t         *     var mode = CryptoJS.mode.CBC.Encryptor.create(cipher, iv.words);\n\t         */\n\t        init: function (cipher, iv) {\n\t            this._cipher = cipher;\n\t            this._iv = iv;\n\t        }\n\t    });\n\n\t    /**\n\t     * Cipher Block Chaining mode.\n\t     */\n\t    var CBC = C_mode.CBC = (function () {\n\t        /**\n\t         * Abstract base CBC mode.\n\t         */\n\t        var CBC = BlockCipherMode.extend();\n\n\t        /**\n\t         * CBC encryptor.\n\t         */\n\t        CBC.Encryptor = CBC.extend({\n\t            /**\n\t             * Processes the data block at offset.\n\t             *\n\t             * @param {Array} words The data words to operate on.\n\t             * @param {number} offset The offset where the block starts.\n\t             *\n\t             * @example\n\t             *\n\t             *     mode.processBlock(data.words, offset);\n\t             */\n\t            processBlock: function (words, offset) {\n\t                // Shortcuts\n\t                var cipher = this._cipher;\n\t                var blockSize = cipher.blockSize;\n\n\t                // XOR and encrypt\n\t                xorBlock.call(this, words, offset, blockSize);\n\t                cipher.encryptBlock(words, offset);\n\n\t                // Remember this block to use with next block\n\t                this._prevBlock = words.slice(offset, offset + blockSize);\n\t            }\n\t        });\n\n\t        /**\n\t         * CBC decryptor.\n\t         */\n\t        CBC.Decryptor = CBC.extend({\n\t            /**\n\t             * Processes the data block at offset.\n\t             *\n\t             * @param {Array} words The data words to operate on.\n\t             * @param {number} offset The offset where the block starts.\n\t             *\n\t             * @example\n\t             *\n\t             *     mode.processBlock(data.words, offset);\n\t             */\n\t            processBlock: function (words, offset) {\n\t                // Shortcuts\n\t                var cipher = this._cipher;\n\t                var blockSize = cipher.blockSize;\n\n\t                // Remember this block to use with next block\n\t                var thisBlock = words.slice(offset, offset + blockSize);\n\n\t                // Decrypt and XOR\n\t                cipher.decryptBlock(words, offset);\n\t                xorBlock.call(this, words, offset, blockSize);\n\n\t                // This block becomes the previous block\n\t                this._prevBlock = thisBlock;\n\t            }\n\t        });\n\n\t        function xorBlock(words, offset, blockSize) {\n\t            // Shortcut\n\t            var iv = this._iv;\n\n\t            // Choose mixing block\n\t            if (iv) {\n\t                var block = iv;\n\n\t                // Remove IV for subsequent blocks\n\t                this._iv = undefined;\n\t            } else {\n\t                var block = this._prevBlock;\n\t            }\n\n\t            // XOR blocks\n\t            for (var i = 0; i < blockSize; i++) {\n\t                words[offset + i] ^= block[i];\n\t            }\n\t        }\n\n\t        return CBC;\n\t    }());\n\n\t    /**\n\t     * Padding namespace.\n\t     */\n\t    var C_pad = C.pad = {};\n\n\t    /**\n\t     * PKCS #5/7 padding strategy.\n\t     */\n\t    var Pkcs7 = C_pad.Pkcs7 = {\n\t        /**\n\t         * Pads data using the algorithm defined in PKCS #5/7.\n\t         *\n\t         * @param {WordArray} data The data to pad.\n\t         * @param {number} blockSize The multiple that the data should be padded to.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     CryptoJS.pad.Pkcs7.pad(wordArray, 4);\n\t         */\n\t        pad: function (data, blockSize) {\n\t            // Shortcut\n\t            var blockSizeBytes = blockSize * 4;\n\n\t            // Count padding bytes\n\t            var nPaddingBytes = blockSizeBytes - data.sigBytes % blockSizeBytes;\n\n\t            // Create padding word\n\t            var paddingWord = (nPaddingBytes << 24) | (nPaddingBytes << 16) | (nPaddingBytes << 8) | nPaddingBytes;\n\n\t            // Create padding\n\t            var paddingWords = [];\n\t            for (var i = 0; i < nPaddingBytes; i += 4) {\n\t                paddingWords.push(paddingWord);\n\t            }\n\t            var padding = WordArray.create(paddingWords, nPaddingBytes);\n\n\t            // Add padding\n\t            data.concat(padding);\n\t        },\n\n\t        /**\n\t         * Unpads data that had been padded using the algorithm defined in PKCS #5/7.\n\t         *\n\t         * @param {WordArray} data The data to unpad.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     CryptoJS.pad.Pkcs7.unpad(wordArray);\n\t         */\n\t        unpad: function (data) {\n\t            // Get number of padding bytes from last byte\n\t            var nPaddingBytes = data.words[(data.sigBytes - 1) >>> 2] & 0xff;\n\n\t            // Remove padding\n\t            data.sigBytes -= nPaddingBytes;\n\t        }\n\t    };\n\n\t    /**\n\t     * Abstract base block cipher template.\n\t     *\n\t     * @property {number} blockSize The number of 32-bit words this cipher operates on. Default: 4 (128 bits)\n\t     */\n\t    var BlockCipher = C_lib.BlockCipher = Cipher.extend({\n\t        /**\n\t         * Configuration options.\n\t         *\n\t         * @property {Mode} mode The block mode to use. Default: CBC\n\t         * @property {Padding} padding The padding strategy to use. Default: Pkcs7\n\t         */\n\t        cfg: Cipher.cfg.extend({\n\t            mode: CBC,\n\t            padding: Pkcs7\n\t        }),\n\n\t        reset: function () {\n\t            // Reset cipher\n\t            Cipher.reset.call(this);\n\n\t            // Shortcuts\n\t            var cfg = this.cfg;\n\t            var iv = cfg.iv;\n\t            var mode = cfg.mode;\n\n\t            // Reset block mode\n\t            if (this._xformMode == this._ENC_XFORM_MODE) {\n\t                var modeCreator = mode.createEncryptor;\n\t            } else /* if (this._xformMode == this._DEC_XFORM_MODE) */ {\n\t                var modeCreator = mode.createDecryptor;\n\t                // Keep at least one block in the buffer for unpadding\n\t                this._minBufferSize = 1;\n\t            }\n\n\t            if (this._mode && this._mode.__creator == modeCreator) {\n\t                this._mode.init(this, iv && iv.words);\n\t            } else {\n\t                this._mode = modeCreator.call(mode, this, iv && iv.words);\n\t                this._mode.__creator = modeCreator;\n\t            }\n\t        },\n\n\t        _doProcessBlock: function (words, offset) {\n\t            this._mode.processBlock(words, offset);\n\t        },\n\n\t        _doFinalize: function () {\n\t            // Shortcut\n\t            var padding = this.cfg.padding;\n\n\t            // Finalize\n\t            if (this._xformMode == this._ENC_XFORM_MODE) {\n\t                // Pad data\n\t                padding.pad(this._data, this.blockSize);\n\n\t                // Process final blocks\n\t                var finalProcessedBlocks = this._process(!!'flush');\n\t            } else /* if (this._xformMode == this._DEC_XFORM_MODE) */ {\n\t                // Process final blocks\n\t                var finalProcessedBlocks = this._process(!!'flush');\n\n\t                // Unpad data\n\t                padding.unpad(finalProcessedBlocks);\n\t            }\n\n\t            return finalProcessedBlocks;\n\t        },\n\n\t        blockSize: 128/32\n\t    });\n\n\t    /**\n\t     * A collection of cipher parameters.\n\t     *\n\t     * @property {WordArray} ciphertext The raw ciphertext.\n\t     * @property {WordArray} key The key to this ciphertext.\n\t     * @property {WordArray} iv The IV used in the ciphering operation.\n\t     * @property {WordArray} salt The salt used with a key derivation function.\n\t     * @property {Cipher} algorithm The cipher algorithm.\n\t     * @property {Mode} mode The block mode used in the ciphering operation.\n\t     * @property {Padding} padding The padding scheme used in the ciphering operation.\n\t     * @property {number} blockSize The block size of the cipher.\n\t     * @property {Format} formatter The default formatting strategy to convert this cipher params object to a string.\n\t     */\n\t    var CipherParams = C_lib.CipherParams = Base.extend({\n\t        /**\n\t         * Initializes a newly created cipher params object.\n\t         *\n\t         * @param {Object} cipherParams An object with any of the possible cipher parameters.\n\t         *\n\t         * @example\n\t         *\n\t         *     var cipherParams = CryptoJS.lib.CipherParams.create({\n\t         *         ciphertext: ciphertextWordArray,\n\t         *         key: keyWordArray,\n\t         *         iv: ivWordArray,\n\t         *         salt: saltWordArray,\n\t         *         algorithm: CryptoJS.algo.AES,\n\t         *         mode: CryptoJS.mode.CBC,\n\t         *         padding: CryptoJS.pad.PKCS7,\n\t         *         blockSize: 4,\n\t         *         formatter: CryptoJS.format.OpenSSL\n\t         *     });\n\t         */\n\t        init: function (cipherParams) {\n\t            this.mixIn(cipherParams);\n\t        },\n\n\t        /**\n\t         * Converts this cipher params object to a string.\n\t         *\n\t         * @param {Format} formatter (Optional) The formatting strategy to use.\n\t         *\n\t         * @return {string} The stringified cipher params.\n\t         *\n\t         * @throws Error If neither the formatter nor the default formatter is set.\n\t         *\n\t         * @example\n\t         *\n\t         *     var string = cipherParams + '';\n\t         *     var string = cipherParams.toString();\n\t         *     var string = cipherParams.toString(CryptoJS.format.OpenSSL);\n\t         */\n\t        toString: function (formatter) {\n\t            return (formatter || this.formatter).stringify(this);\n\t        }\n\t    });\n\n\t    /**\n\t     * Format namespace.\n\t     */\n\t    var C_format = C.format = {};\n\n\t    /**\n\t     * OpenSSL formatting strategy.\n\t     */\n\t    var OpenSSLFormatter = C_format.OpenSSL = {\n\t        /**\n\t         * Converts a cipher params object to an OpenSSL-compatible string.\n\t         *\n\t         * @param {CipherParams} cipherParams The cipher params object.\n\t         *\n\t         * @return {string} The OpenSSL-compatible string.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var openSSLString = CryptoJS.format.OpenSSL.stringify(cipherParams);\n\t         */\n\t        stringify: function (cipherParams) {\n\t            // Shortcuts\n\t            var ciphertext = cipherParams.ciphertext;\n\t            var salt = cipherParams.salt;\n\n\t            // Format\n\t            if (salt) {\n\t                var wordArray = WordArray.create([0x53616c74, 0x65645f5f]).concat(salt).concat(ciphertext);\n\t            } else {\n\t                var wordArray = ciphertext;\n\t            }\n\n\t            return wordArray.toString(Base64);\n\t        },\n\n\t        /**\n\t         * Converts an OpenSSL-compatible string to a cipher params object.\n\t         *\n\t         * @param {string} openSSLStr The OpenSSL-compatible string.\n\t         *\n\t         * @return {CipherParams} The cipher params object.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var cipherParams = CryptoJS.format.OpenSSL.parse(openSSLString);\n\t         */\n\t        parse: function (openSSLStr) {\n\t            // Parse base64\n\t            var ciphertext = Base64.parse(openSSLStr);\n\n\t            // Shortcut\n\t            var ciphertextWords = ciphertext.words;\n\n\t            // Test for salt\n\t            if (ciphertextWords[0] == 0x53616c74 && ciphertextWords[1] == 0x65645f5f) {\n\t                // Extract salt\n\t                var salt = WordArray.create(ciphertextWords.slice(2, 4));\n\n\t                // Remove salt from ciphertext\n\t                ciphertextWords.splice(0, 4);\n\t                ciphertext.sigBytes -= 16;\n\t            }\n\n\t            return CipherParams.create({ ciphertext: ciphertext, salt: salt });\n\t        }\n\t    };\n\n\t    /**\n\t     * A cipher wrapper that returns ciphertext as a serializable cipher params object.\n\t     */\n\t    var SerializableCipher = C_lib.SerializableCipher = Base.extend({\n\t        /**\n\t         * Configuration options.\n\t         *\n\t         * @property {Formatter} format The formatting strategy to convert cipher param objects to and from a string. Default: OpenSSL\n\t         */\n\t        cfg: Base.extend({\n\t            format: OpenSSLFormatter\n\t        }),\n\n\t        /**\n\t         * Encrypts a message.\n\t         *\n\t         * @param {Cipher} cipher The cipher algorithm to use.\n\t         * @param {WordArray|string} message The message to encrypt.\n\t         * @param {WordArray} key The key.\n\t         * @param {Object} cfg (Optional) The configuration options to use for this operation.\n\t         *\n\t         * @return {CipherParams} A cipher params object.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key);\n\t         *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key, { iv: iv });\n\t         *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key, { iv: iv, format: CryptoJS.format.OpenSSL });\n\t         */\n\t        encrypt: function (cipher, message, key, cfg) {\n\t            // Apply config defaults\n\t            cfg = this.cfg.extend(cfg);\n\n\t            // Encrypt\n\t            var encryptor = cipher.createEncryptor(key, cfg);\n\t            var ciphertext = encryptor.finalize(message);\n\n\t            // Shortcut\n\t            var cipherCfg = encryptor.cfg;\n\n\t            // Create and return serializable cipher params\n\t            return CipherParams.create({\n\t                ciphertext: ciphertext,\n\t                key: key,\n\t                iv: cipherCfg.iv,\n\t                algorithm: cipher,\n\t                mode: cipherCfg.mode,\n\t                padding: cipherCfg.padding,\n\t                blockSize: cipher.blockSize,\n\t                formatter: cfg.format\n\t            });\n\t        },\n\n\t        /**\n\t         * Decrypts serialized ciphertext.\n\t         *\n\t         * @param {Cipher} cipher The cipher algorithm to use.\n\t         * @param {CipherParams|string} ciphertext The ciphertext to decrypt.\n\t         * @param {WordArray} key The key.\n\t         * @param {Object} cfg (Optional) The configuration options to use for this operation.\n\t         *\n\t         * @return {WordArray} The plaintext.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var plaintext = CryptoJS.lib.SerializableCipher.decrypt(CryptoJS.algo.AES, formattedCiphertext, key, { iv: iv, format: CryptoJS.format.OpenSSL });\n\t         *     var plaintext = CryptoJS.lib.SerializableCipher.decrypt(CryptoJS.algo.AES, ciphertextParams, key, { iv: iv, format: CryptoJS.format.OpenSSL });\n\t         */\n\t        decrypt: function (cipher, ciphertext, key, cfg) {\n\t            // Apply config defaults\n\t            cfg = this.cfg.extend(cfg);\n\n\t            // Convert string to CipherParams\n\t            ciphertext = this._parse(ciphertext, cfg.format);\n\n\t            // Decrypt\n\t            var plaintext = cipher.createDecryptor(key, cfg).finalize(ciphertext.ciphertext);\n\n\t            return plaintext;\n\t        },\n\n\t        /**\n\t         * Converts serialized ciphertext to CipherParams,\n\t         * else assumed CipherParams already and returns ciphertext unchanged.\n\t         *\n\t         * @param {CipherParams|string} ciphertext The ciphertext.\n\t         * @param {Formatter} format The formatting strategy to use to parse serialized ciphertext.\n\t         *\n\t         * @return {CipherParams} The unserialized ciphertext.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var ciphertextParams = CryptoJS.lib.SerializableCipher._parse(ciphertextStringOrParams, format);\n\t         */\n\t        _parse: function (ciphertext, format) {\n\t            if (typeof ciphertext == 'string') {\n\t                return format.parse(ciphertext, this);\n\t            } else {\n\t                return ciphertext;\n\t            }\n\t        }\n\t    });\n\n\t    /**\n\t     * Key derivation function namespace.\n\t     */\n\t    var C_kdf = C.kdf = {};\n\n\t    /**\n\t     * OpenSSL key derivation function.\n\t     */\n\t    var OpenSSLKdf = C_kdf.OpenSSL = {\n\t        /**\n\t         * Derives a key and IV from a password.\n\t         *\n\t         * @param {string} password The password to derive from.\n\t         * @param {number} keySize The size in words of the key to generate.\n\t         * @param {number} ivSize The size in words of the IV to generate.\n\t         * @param {WordArray|string} salt (Optional) A 64-bit salt to use. If omitted, a salt will be generated randomly.\n\t         *\n\t         * @return {CipherParams} A cipher params object with the key, IV, and salt.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var derivedParams = CryptoJS.kdf.OpenSSL.execute('Password', 256/32, 128/32);\n\t         *     var derivedParams = CryptoJS.kdf.OpenSSL.execute('Password', 256/32, 128/32, 'saltsalt');\n\t         */\n\t        execute: function (password, keySize, ivSize, salt) {\n\t            // Generate random salt\n\t            if (!salt) {\n\t                salt = WordArray.random(64/8);\n\t            }\n\n\t            // Derive key and IV\n\t            var key = EvpKDF.create({ keySize: keySize + ivSize }).compute(password, salt);\n\n\t            // Separate key and IV\n\t            var iv = WordArray.create(key.words.slice(keySize), ivSize * 4);\n\t            key.sigBytes = keySize * 4;\n\n\t            // Return params\n\t            return CipherParams.create({ key: key, iv: iv, salt: salt });\n\t        }\n\t    };\n\n\t    /**\n\t     * A serializable cipher wrapper that derives the key from a password,\n\t     * and returns ciphertext as a serializable cipher params object.\n\t     */\n\t    var PasswordBasedCipher = C_lib.PasswordBasedCipher = SerializableCipher.extend({\n\t        /**\n\t         * Configuration options.\n\t         *\n\t         * @property {KDF} kdf The key derivation function to use to generate a key and IV from a password. Default: OpenSSL\n\t         */\n\t        cfg: SerializableCipher.cfg.extend({\n\t            kdf: OpenSSLKdf\n\t        }),\n\n\t        /**\n\t         * Encrypts a message using a password.\n\t         *\n\t         * @param {Cipher} cipher The cipher algorithm to use.\n\t         * @param {WordArray|string} message The message to encrypt.\n\t         * @param {string} password The password.\n\t         * @param {Object} cfg (Optional) The configuration options to use for this operation.\n\t         *\n\t         * @return {CipherParams} A cipher params object.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var ciphertextParams = CryptoJS.lib.PasswordBasedCipher.encrypt(CryptoJS.algo.AES, message, 'password');\n\t         *     var ciphertextParams = CryptoJS.lib.PasswordBasedCipher.encrypt(CryptoJS.algo.AES, message, 'password', { format: CryptoJS.format.OpenSSL });\n\t         */\n\t        encrypt: function (cipher, message, password, cfg) {\n\t            // Apply config defaults\n\t            cfg = this.cfg.extend(cfg);\n\n\t            // Derive key and other params\n\t            var derivedParams = cfg.kdf.execute(password, cipher.keySize, cipher.ivSize);\n\n\t            // Add IV to config\n\t            cfg.iv = derivedParams.iv;\n\n\t            // Encrypt\n\t            var ciphertext = SerializableCipher.encrypt.call(this, cipher, message, derivedParams.key, cfg);\n\n\t            // Mix in derived params\n\t            ciphertext.mixIn(derivedParams);\n\n\t            return ciphertext;\n\t        },\n\n\t        /**\n\t         * Decrypts serialized ciphertext using a password.\n\t         *\n\t         * @param {Cipher} cipher The cipher algorithm to use.\n\t         * @param {CipherParams|string} ciphertext The ciphertext to decrypt.\n\t         * @param {string} password The password.\n\t         * @param {Object} cfg (Optional) The configuration options to use for this operation.\n\t         *\n\t         * @return {WordArray} The plaintext.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var plaintext = CryptoJS.lib.PasswordBasedCipher.decrypt(CryptoJS.algo.AES, formattedCiphertext, 'password', { format: CryptoJS.format.OpenSSL });\n\t         *     var plaintext = CryptoJS.lib.PasswordBasedCipher.decrypt(CryptoJS.algo.AES, ciphertextParams, 'password', { format: CryptoJS.format.OpenSSL });\n\t         */\n\t        decrypt: function (cipher, ciphertext, password, cfg) {\n\t            // Apply config defaults\n\t            cfg = this.cfg.extend(cfg);\n\n\t            // Convert string to CipherParams\n\t            ciphertext = this._parse(ciphertext, cfg.format);\n\n\t            // Derive key and other params\n\t            var derivedParams = cfg.kdf.execute(password, cipher.keySize, cipher.ivSize, ciphertext.salt);\n\n\t            // Add IV to config\n\t            cfg.iv = derivedParams.iv;\n\n\t            // Decrypt\n\t            var plaintext = SerializableCipher.decrypt.call(this, cipher, ciphertext, derivedParams.key, cfg);\n\n\t            return plaintext;\n\t        }\n\t    });\n\t}());\n\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory();\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\troot.CryptoJS = factory();\n\t}\n}(this, function () {\n\n\t/**\n\t * CryptoJS core components.\n\t */\n\tvar CryptoJS = CryptoJS || (function (Math, undefined) {\n\t    /*\n\t     * Local polyfil of Object.create\n\t     */\n\t    var create = Object.create || (function () {\n\t        function F() {};\n\n\t        return function (obj) {\n\t            var subtype;\n\n\t            F.prototype = obj;\n\n\t            subtype = new F();\n\n\t            F.prototype = null;\n\n\t            return subtype;\n\t        };\n\t    }())\n\n\t    /**\n\t     * CryptoJS namespace.\n\t     */\n\t    var C = {};\n\n\t    /**\n\t     * Library namespace.\n\t     */\n\t    var C_lib = C.lib = {};\n\n\t    /**\n\t     * Base object for prototypal inheritance.\n\t     */\n\t    var Base = C_lib.Base = (function () {\n\n\n\t        return {\n\t            /**\n\t             * Creates a new object that inherits from this object.\n\t             *\n\t             * @param {Object} overrides Properties to copy into the new object.\n\t             *\n\t             * @return {Object} The new object.\n\t             *\n\t             * @static\n\t             *\n\t             * @example\n\t             *\n\t             *     var MyType = CryptoJS.lib.Base.extend({\n\t             *         field: 'value',\n\t             *\n\t             *         method: function () {\n\t             *         }\n\t             *     });\n\t             */\n\t            extend: function (overrides) {\n\t                // Spawn\n\t                var subtype = create(this);\n\n\t                // Augment\n\t                if (overrides) {\n\t                    subtype.mixIn(overrides);\n\t                }\n\n\t                // Create default initializer\n\t                if (!subtype.hasOwnProperty('init') || this.init === subtype.init) {\n\t                    subtype.init = function () {\n\t                        subtype.$super.init.apply(this, arguments);\n\t                    };\n\t                }\n\n\t                // Initializer's prototype is the subtype object\n\t                subtype.init.prototype = subtype;\n\n\t                // Reference supertype\n\t                subtype.$super = this;\n\n\t                return subtype;\n\t            },\n\n\t            /**\n\t             * Extends this object and runs the init method.\n\t             * Arguments to create() will be passed to init().\n\t             *\n\t             * @return {Object} The new object.\n\t             *\n\t             * @static\n\t             *\n\t             * @example\n\t             *\n\t             *     var instance = MyType.create();\n\t             */\n\t            create: function () {\n\t                var instance = this.extend();\n\t                instance.init.apply(instance, arguments);\n\n\t                return instance;\n\t            },\n\n\t            /**\n\t             * Initializes a newly created object.\n\t             * Override this method to add some logic when your objects are created.\n\t             *\n\t             * @example\n\t             *\n\t             *     var MyType = CryptoJS.lib.Base.extend({\n\t             *         init: function () {\n\t             *             // ...\n\t             *         }\n\t             *     });\n\t             */\n\t            init: function () {\n\t            },\n\n\t            /**\n\t             * Copies properties into this object.\n\t             *\n\t             * @param {Object} properties The properties to mix in.\n\t             *\n\t             * @example\n\t             *\n\t             *     MyType.mixIn({\n\t             *         field: 'value'\n\t             *     });\n\t             */\n\t            mixIn: function (properties) {\n\t                for (var propertyName in properties) {\n\t                    if (properties.hasOwnProperty(propertyName)) {\n\t                        this[propertyName] = properties[propertyName];\n\t                    }\n\t                }\n\n\t                // IE won't copy toString using the loop above\n\t                if (properties.hasOwnProperty('toString')) {\n\t                    this.toString = properties.toString;\n\t                }\n\t            },\n\n\t            /**\n\t             * Creates a copy of this object.\n\t             *\n\t             * @return {Object} The clone.\n\t             *\n\t             * @example\n\t             *\n\t             *     var clone = instance.clone();\n\t             */\n\t            clone: function () {\n\t                return this.init.prototype.extend(this);\n\t            }\n\t        };\n\t    }());\n\n\t    /**\n\t     * An array of 32-bit words.\n\t     *\n\t     * @property {Array} words The array of 32-bit words.\n\t     * @property {number} sigBytes The number of significant bytes in this word array.\n\t     */\n\t    var WordArray = C_lib.WordArray = Base.extend({\n\t        /**\n\t         * Initializes a newly created word array.\n\t         *\n\t         * @param {Array} words (Optional) An array of 32-bit words.\n\t         * @param {number} sigBytes (Optional) The number of significant bytes in the words.\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.lib.WordArray.create();\n\t         *     var wordArray = CryptoJS.lib.WordArray.create([0x00010203, 0x04050607]);\n\t         *     var wordArray = CryptoJS.lib.WordArray.create([0x00010203, 0x04050607], 6);\n\t         */\n\t        init: function (words, sigBytes) {\n\t            words = this.words = words || [];\n\n\t            if (sigBytes != undefined) {\n\t                this.sigBytes = sigBytes;\n\t            } else {\n\t                this.sigBytes = words.length * 4;\n\t            }\n\t        },\n\n\t        /**\n\t         * Converts this word array to a string.\n\t         *\n\t         * @param {Encoder} encoder (Optional) The encoding strategy to use. Default: CryptoJS.enc.Hex\n\t         *\n\t         * @return {string} The stringified word array.\n\t         *\n\t         * @example\n\t         *\n\t         *     var string = wordArray + '';\n\t         *     var string = wordArray.toString();\n\t         *     var string = wordArray.toString(CryptoJS.enc.Utf8);\n\t         */\n\t        toString: function (encoder) {\n\t            return (encoder || Hex).stringify(this);\n\t        },\n\n\t        /**\n\t         * Concatenates a word array to this word array.\n\t         *\n\t         * @param {WordArray} wordArray The word array to append.\n\t         *\n\t         * @return {WordArray} This word array.\n\t         *\n\t         * @example\n\t         *\n\t         *     wordArray1.concat(wordArray2);\n\t         */\n\t        concat: function (wordArray) {\n\t            // Shortcuts\n\t            var thisWords = this.words;\n\t            var thatWords = wordArray.words;\n\t            var thisSigBytes = this.sigBytes;\n\t            var thatSigBytes = wordArray.sigBytes;\n\n\t            // Clamp excess bits\n\t            this.clamp();\n\n\t            // Concat\n\t            if (thisSigBytes % 4) {\n\t                // Copy one byte at a time\n\t                for (var i = 0; i < thatSigBytes; i++) {\n\t                    var thatByte = (thatWords[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;\n\t                    thisWords[(thisSigBytes + i) >>> 2] |= thatByte << (24 - ((thisSigBytes + i) % 4) * 8);\n\t                }\n\t            } else {\n\t                // Copy one word at a time\n\t                for (var i = 0; i < thatSigBytes; i += 4) {\n\t                    thisWords[(thisSigBytes + i) >>> 2] = thatWords[i >>> 2];\n\t                }\n\t            }\n\t            this.sigBytes += thatSigBytes;\n\n\t            // Chainable\n\t            return this;\n\t        },\n\n\t        /**\n\t         * Removes insignificant bits.\n\t         *\n\t         * @example\n\t         *\n\t         *     wordArray.clamp();\n\t         */\n\t        clamp: function () {\n\t            // Shortcuts\n\t            var words = this.words;\n\t            var sigBytes = this.sigBytes;\n\n\t            // Clamp\n\t            words[sigBytes >>> 2] &= 0xffffffff << (32 - (sigBytes % 4) * 8);\n\t            words.length = Math.ceil(sigBytes / 4);\n\t        },\n\n\t        /**\n\t         * Creates a copy of this word array.\n\t         *\n\t         * @return {WordArray} The clone.\n\t         *\n\t         * @example\n\t         *\n\t         *     var clone = wordArray.clone();\n\t         */\n\t        clone: function () {\n\t            var clone = Base.clone.call(this);\n\t            clone.words = this.words.slice(0);\n\n\t            return clone;\n\t        },\n\n\t        /**\n\t         * Creates a word array filled with random bytes.\n\t         *\n\t         * @param {number} nBytes The number of random bytes to generate.\n\t         *\n\t         * @return {WordArray} The random word array.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.lib.WordArray.random(16);\n\t         */\n\t        random: function (nBytes) {\n\t            var words = [];\n\n\t            var r = (function (m_w) {\n\t                var m_w = m_w;\n\t                var m_z = 0x3ade68b1;\n\t                var mask = 0xffffffff;\n\n\t                return function () {\n\t                    m_z = (0x9069 * (m_z & 0xFFFF) + (m_z >> 0x10)) & mask;\n\t                    m_w = (0x4650 * (m_w & 0xFFFF) + (m_w >> 0x10)) & mask;\n\t                    var result = ((m_z << 0x10) + m_w) & mask;\n\t                    result /= 0x100000000;\n\t                    result += 0.5;\n\t                    return result * (Math.random() > .5 ? 1 : -1);\n\t                }\n\t            });\n\n\t            for (var i = 0, rcache; i < nBytes; i += 4) {\n\t                var _r = r((rcache || Math.random()) * 0x100000000);\n\n\t                rcache = _r() * 0x3ade67b7;\n\t                words.push((_r() * 0x100000000) | 0);\n\t            }\n\n\t            return new WordArray.init(words, nBytes);\n\t        }\n\t    });\n\n\t    /**\n\t     * Encoder namespace.\n\t     */\n\t    var C_enc = C.enc = {};\n\n\t    /**\n\t     * Hex encoding strategy.\n\t     */\n\t    var Hex = C_enc.Hex = {\n\t        /**\n\t         * Converts a word array to a hex string.\n\t         *\n\t         * @param {WordArray} wordArray The word array.\n\t         *\n\t         * @return {string} The hex string.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var hexString = CryptoJS.enc.Hex.stringify(wordArray);\n\t         */\n\t        stringify: function (wordArray) {\n\t            // Shortcuts\n\t            var words = wordArray.words;\n\t            var sigBytes = wordArray.sigBytes;\n\n\t            // Convert\n\t            var hexChars = [];\n\t            for (var i = 0; i < sigBytes; i++) {\n\t                var bite = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;\n\t                hexChars.push((bite >>> 4).toString(16));\n\t                hexChars.push((bite & 0x0f).toString(16));\n\t            }\n\n\t            return hexChars.join('');\n\t        },\n\n\t        /**\n\t         * Converts a hex string to a word array.\n\t         *\n\t         * @param {string} hexStr The hex string.\n\t         *\n\t         * @return {WordArray} The word array.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.enc.Hex.parse(hexString);\n\t         */\n\t        parse: function (hexStr) {\n\t            // Shortcut\n\t            var hexStrLength = hexStr.length;\n\n\t            // Convert\n\t            var words = [];\n\t            for (var i = 0; i < hexStrLength; i += 2) {\n\t                words[i >>> 3] |= parseInt(hexStr.substr(i, 2), 16) << (24 - (i % 8) * 4);\n\t            }\n\n\t            return new WordArray.init(words, hexStrLength / 2);\n\t        }\n\t    };\n\n\t    /**\n\t     * Latin1 encoding strategy.\n\t     */\n\t    var Latin1 = C_enc.Latin1 = {\n\t        /**\n\t         * Converts a word array to a Latin1 string.\n\t         *\n\t         * @param {WordArray} wordArray The word array.\n\t         *\n\t         * @return {string} The Latin1 string.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var latin1String = CryptoJS.enc.Latin1.stringify(wordArray);\n\t         */\n\t        stringify: function (wordArray) {\n\t            // Shortcuts\n\t            var words = wordArray.words;\n\t            var sigBytes = wordArray.sigBytes;\n\n\t            // Convert\n\t            var latin1Chars = [];\n\t            for (var i = 0; i < sigBytes; i++) {\n\t                var bite = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;\n\t                latin1Chars.push(String.fromCharCode(bite));\n\t            }\n\n\t            return latin1Chars.join('');\n\t        },\n\n\t        /**\n\t         * Converts a Latin1 string to a word array.\n\t         *\n\t         * @param {string} latin1Str The Latin1 string.\n\t         *\n\t         * @return {WordArray} The word array.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.enc.Latin1.parse(latin1String);\n\t         */\n\t        parse: function (latin1Str) {\n\t            // Shortcut\n\t            var latin1StrLength = latin1Str.length;\n\n\t            // Convert\n\t            var words = [];\n\t            for (var i = 0; i < latin1StrLength; i++) {\n\t                words[i >>> 2] |= (latin1Str.charCodeAt(i) & 0xff) << (24 - (i % 4) * 8);\n\t            }\n\n\t            return new WordArray.init(words, latin1StrLength);\n\t        }\n\t    };\n\n\t    /**\n\t     * UTF-8 encoding strategy.\n\t     */\n\t    var Utf8 = C_enc.Utf8 = {\n\t        /**\n\t         * Converts a word array to a UTF-8 string.\n\t         *\n\t         * @param {WordArray} wordArray The word array.\n\t         *\n\t         * @return {string} The UTF-8 string.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var utf8String = CryptoJS.enc.Utf8.stringify(wordArray);\n\t         */\n\t        stringify: function (wordArray) {\n\t            try {\n\t                return decodeURIComponent(escape(Latin1.stringify(wordArray)));\n\t            } catch (e) {\n\t                throw new Error('Malformed UTF-8 data');\n\t            }\n\t        },\n\n\t        /**\n\t         * Converts a UTF-8 string to a word array.\n\t         *\n\t         * @param {string} utf8Str The UTF-8 string.\n\t         *\n\t         * @return {WordArray} The word array.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.enc.Utf8.parse(utf8String);\n\t         */\n\t        parse: function (utf8Str) {\n\t            return Latin1.parse(unescape(encodeURIComponent(utf8Str)));\n\t        }\n\t    };\n\n\t    /**\n\t     * Abstract buffered block algorithm template.\n\t     *\n\t     * The property blockSize must be implemented in a concrete subtype.\n\t     *\n\t     * @property {number} _minBufferSize The number of blocks that should be kept unprocessed in the buffer. Default: 0\n\t     */\n\t    var BufferedBlockAlgorithm = C_lib.BufferedBlockAlgorithm = Base.extend({\n\t        /**\n\t         * Resets this block algorithm's data buffer to its initial state.\n\t         *\n\t         * @example\n\t         *\n\t         *     bufferedBlockAlgorithm.reset();\n\t         */\n\t        reset: function () {\n\t            // Initial values\n\t            this._data = new WordArray.init();\n\t            this._nDataBytes = 0;\n\t        },\n\n\t        /**\n\t         * Adds new data to this block algorithm's buffer.\n\t         *\n\t         * @param {WordArray|string} data The data to append. Strings are converted to a WordArray using UTF-8.\n\t         *\n\t         * @example\n\t         *\n\t         *     bufferedBlockAlgorithm._append('data');\n\t         *     bufferedBlockAlgorithm._append(wordArray);\n\t         */\n\t        _append: function (data) {\n\t            // Convert string to WordArray, else assume WordArray already\n\t            if (typeof data == 'string') {\n\t                data = Utf8.parse(data);\n\t            }\n\n\t            // Append\n\t            this._data.concat(data);\n\t            this._nDataBytes += data.sigBytes;\n\t        },\n\n\t        /**\n\t         * Processes available data blocks.\n\t         *\n\t         * This method invokes _doProcessBlock(offset), which must be implemented by a concrete subtype.\n\t         *\n\t         * @param {boolean} doFlush Whether all blocks and partial blocks should be processed.\n\t         *\n\t         * @return {WordArray} The processed data.\n\t         *\n\t         * @example\n\t         *\n\t         *     var processedData = bufferedBlockAlgorithm._process();\n\t         *     var processedData = bufferedBlockAlgorithm._process(!!'flush');\n\t         */\n\t        _process: function (doFlush) {\n\t            // Shortcuts\n\t            var data = this._data;\n\t            var dataWords = data.words;\n\t            var dataSigBytes = data.sigBytes;\n\t            var blockSize = this.blockSize;\n\t            var blockSizeBytes = blockSize * 4;\n\n\t            // Count blocks ready\n\t            var nBlocksReady = dataSigBytes / blockSizeBytes;\n\t            if (doFlush) {\n\t                // Round up to include partial blocks\n\t                nBlocksReady = Math.ceil(nBlocksReady);\n\t            } else {\n\t                // Round down to include only full blocks,\n\t                // less the number of blocks that must remain in the buffer\n\t                nBlocksReady = Math.max((nBlocksReady | 0) - this._minBufferSize, 0);\n\t            }\n\n\t            // Count words ready\n\t            var nWordsReady = nBlocksReady * blockSize;\n\n\t            // Count bytes ready\n\t            var nBytesReady = Math.min(nWordsReady * 4, dataSigBytes);\n\n\t            // Process blocks\n\t            if (nWordsReady) {\n\t                for (var offset = 0; offset < nWordsReady; offset += blockSize) {\n\t                    // Perform concrete-algorithm logic\n\t                    this._doProcessBlock(dataWords, offset);\n\t                }\n\n\t                // Remove processed words\n\t                var processedWords = dataWords.splice(0, nWordsReady);\n\t                data.sigBytes -= nBytesReady;\n\t            }\n\n\t            // Return processed words\n\t            return new WordArray.init(processedWords, nBytesReady);\n\t        },\n\n\t        /**\n\t         * Creates a copy of this object.\n\t         *\n\t         * @return {Object} The clone.\n\t         *\n\t         * @example\n\t         *\n\t         *     var clone = bufferedBlockAlgorithm.clone();\n\t         */\n\t        clone: function () {\n\t            var clone = Base.clone.call(this);\n\t            clone._data = this._data.clone();\n\n\t            return clone;\n\t        },\n\n\t        _minBufferSize: 0\n\t    });\n\n\t    /**\n\t     * Abstract hasher template.\n\t     *\n\t     * @property {number} blockSize The number of 32-bit words this hasher operates on. Default: 16 (512 bits)\n\t     */\n\t    var Hasher = C_lib.Hasher = BufferedBlockAlgorithm.extend({\n\t        /**\n\t         * Configuration options.\n\t         */\n\t        cfg: Base.extend(),\n\n\t        /**\n\t         * Initializes a newly created hasher.\n\t         *\n\t         * @param {Object} cfg (Optional) The configuration options to use for this hash computation.\n\t         *\n\t         * @example\n\t         *\n\t         *     var hasher = CryptoJS.algo.SHA256.create();\n\t         */\n\t        init: function (cfg) {\n\t            // Apply config defaults\n\t            this.cfg = this.cfg.extend(cfg);\n\n\t            // Set initial values\n\t            this.reset();\n\t        },\n\n\t        /**\n\t         * Resets this hasher to its initial state.\n\t         *\n\t         * @example\n\t         *\n\t         *     hasher.reset();\n\t         */\n\t        reset: function () {\n\t            // Reset data buffer\n\t            BufferedBlockAlgorithm.reset.call(this);\n\n\t            // Perform concrete-hasher logic\n\t            this._doReset();\n\t        },\n\n\t        /**\n\t         * Updates this hasher with a message.\n\t         *\n\t         * @param {WordArray|string} messageUpdate The message to append.\n\t         *\n\t         * @return {Hasher} This hasher.\n\t         *\n\t         * @example\n\t         *\n\t         *     hasher.update('message');\n\t         *     hasher.update(wordArray);\n\t         */\n\t        update: function (messageUpdate) {\n\t            // Append\n\t            this._append(messageUpdate);\n\n\t            // Update the hash\n\t            this._process();\n\n\t            // Chainable\n\t            return this;\n\t        },\n\n\t        /**\n\t         * Finalizes the hash computation.\n\t         * Note that the finalize operation is effectively a destructive, read-once operation.\n\t         *\n\t         * @param {WordArray|string} messageUpdate (Optional) A final message update.\n\t         *\n\t         * @return {WordArray} The hash.\n\t         *\n\t         * @example\n\t         *\n\t         *     var hash = hasher.finalize();\n\t         *     var hash = hasher.finalize('message');\n\t         *     var hash = hasher.finalize(wordArray);\n\t         */\n\t        finalize: function (messageUpdate) {\n\t            // Final message update\n\t            if (messageUpdate) {\n\t                this._append(messageUpdate);\n\t            }\n\n\t            // Perform concrete-hasher logic\n\t            var hash = this._doFinalize();\n\n\t            return hash;\n\t        },\n\n\t        blockSize: 512/32,\n\n\t        /**\n\t         * Creates a shortcut function to a hasher's object interface.\n\t         *\n\t         * @param {Hasher} hasher The hasher to create a helper for.\n\t         *\n\t         * @return {Function} The shortcut function.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var SHA256 = CryptoJS.lib.Hasher._createHelper(CryptoJS.algo.SHA256);\n\t         */\n\t        _createHelper: function (hasher) {\n\t            return function (message, cfg) {\n\t                return new hasher.init(cfg).finalize(message);\n\t            };\n\t        },\n\n\t        /**\n\t         * Creates a shortcut function to the HMAC's object interface.\n\t         *\n\t         * @param {Hasher} hasher The hasher to use in this HMAC helper.\n\t         *\n\t         * @return {Function} The shortcut function.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var HmacSHA256 = CryptoJS.lib.Hasher._createHmacHelper(CryptoJS.algo.SHA256);\n\t         */\n\t        _createHmacHelper: function (hasher) {\n\t            return function (message, key) {\n\t                return new C_algo.HMAC.init(hasher, key).finalize(message);\n\t            };\n\t        }\n\t    });\n\n\t    /**\n\t     * Algorithm namespace.\n\t     */\n\t    var C_algo = C.algo = {};\n\n\t    return C;\n\t}(Math));\n\n\n\treturn CryptoJS;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var C_enc = C.enc;\n\n\t    /**\n\t     * Base64 encoding strategy.\n\t     */\n\t    var Base64 = C_enc.Base64 = {\n\t        /**\n\t         * Converts a word array to a Base64 string.\n\t         *\n\t         * @param {WordArray} wordArray The word array.\n\t         *\n\t         * @return {string} The Base64 string.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var base64String = CryptoJS.enc.Base64.stringify(wordArray);\n\t         */\n\t        stringify: function (wordArray) {\n\t            // Shortcuts\n\t            var words = wordArray.words;\n\t            var sigBytes = wordArray.sigBytes;\n\t            var map = this._map;\n\n\t            // Clamp excess bits\n\t            wordArray.clamp();\n\n\t            // Convert\n\t            var base64Chars = [];\n\t            for (var i = 0; i < sigBytes; i += 3) {\n\t                var byte1 = (words[i >>> 2]       >>> (24 - (i % 4) * 8))       & 0xff;\n\t                var byte2 = (words[(i + 1) >>> 2] >>> (24 - ((i + 1) % 4) * 8)) & 0xff;\n\t                var byte3 = (words[(i + 2) >>> 2] >>> (24 - ((i + 2) % 4) * 8)) & 0xff;\n\n\t                var triplet = (byte1 << 16) | (byte2 << 8) | byte3;\n\n\t                for (var j = 0; (j < 4) && (i + j * 0.75 < sigBytes); j++) {\n\t                    base64Chars.push(map.charAt((triplet >>> (6 * (3 - j))) & 0x3f));\n\t                }\n\t            }\n\n\t            // Add padding\n\t            var paddingChar = map.charAt(64);\n\t            if (paddingChar) {\n\t                while (base64Chars.length % 4) {\n\t                    base64Chars.push(paddingChar);\n\t                }\n\t            }\n\n\t            return base64Chars.join('');\n\t        },\n\n\t        /**\n\t         * Converts a Base64 string to a word array.\n\t         *\n\t         * @param {string} base64Str The Base64 string.\n\t         *\n\t         * @return {WordArray} The word array.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.enc.Base64.parse(base64String);\n\t         */\n\t        parse: function (base64Str) {\n\t            // Shortcuts\n\t            var base64StrLength = base64Str.length;\n\t            var map = this._map;\n\t            var reverseMap = this._reverseMap;\n\n\t            if (!reverseMap) {\n\t                    reverseMap = this._reverseMap = [];\n\t                    for (var j = 0; j < map.length; j++) {\n\t                        reverseMap[map.charCodeAt(j)] = j;\n\t                    }\n\t            }\n\n\t            // Ignore padding\n\t            var paddingChar = map.charAt(64);\n\t            if (paddingChar) {\n\t                var paddingIndex = base64Str.indexOf(paddingChar);\n\t                if (paddingIndex !== -1) {\n\t                    base64StrLength = paddingIndex;\n\t                }\n\t            }\n\n\t            // Convert\n\t            return parseLoop(base64Str, base64StrLength, reverseMap);\n\n\t        },\n\n\t        _map: 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/='\n\t    };\n\n\t    function parseLoop(base64Str, base64StrLength, reverseMap) {\n\t      var words = [];\n\t      var nBytes = 0;\n\t      for (var i = 0; i < base64StrLength; i++) {\n\t          if (i % 4) {\n\t              var bits1 = reverseMap[base64Str.charCodeAt(i - 1)] << ((i % 4) * 2);\n\t              var bits2 = reverseMap[base64Str.charCodeAt(i)] >>> (6 - (i % 4) * 2);\n\t              words[nBytes >>> 2] |= (bits1 | bits2) << (24 - (nBytes % 4) * 8);\n\t              nBytes++;\n\t          }\n\t      }\n\t      return WordArray.create(words, nBytes);\n\t    }\n\t}());\n\n\n\treturn CryptoJS.enc.Base64;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var C_enc = C.enc;\n\n\t    /**\n\t     * UTF-16 BE encoding strategy.\n\t     */\n\t    var Utf16BE = C_enc.Utf16 = C_enc.Utf16BE = {\n\t        /**\n\t         * Converts a word array to a UTF-16 BE string.\n\t         *\n\t         * @param {WordArray} wordArray The word array.\n\t         *\n\t         * @return {string} The UTF-16 BE string.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var utf16String = CryptoJS.enc.Utf16.stringify(wordArray);\n\t         */\n\t        stringify: function (wordArray) {\n\t            // Shortcuts\n\t            var words = wordArray.words;\n\t            var sigBytes = wordArray.sigBytes;\n\n\t            // Convert\n\t            var utf16Chars = [];\n\t            for (var i = 0; i < sigBytes; i += 2) {\n\t                var codePoint = (words[i >>> 2] >>> (16 - (i % 4) * 8)) & 0xffff;\n\t                utf16Chars.push(String.fromCharCode(codePoint));\n\t            }\n\n\t            return utf16Chars.join('');\n\t        },\n\n\t        /**\n\t         * Converts a UTF-16 BE string to a word array.\n\t         *\n\t         * @param {string} utf16Str The UTF-16 BE string.\n\t         *\n\t         * @return {WordArray} The word array.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.enc.Utf16.parse(utf16String);\n\t         */\n\t        parse: function (utf16Str) {\n\t            // Shortcut\n\t            var utf16StrLength = utf16Str.length;\n\n\t            // Convert\n\t            var words = [];\n\t            for (var i = 0; i < utf16StrLength; i++) {\n\t                words[i >>> 1] |= utf16Str.charCodeAt(i) << (16 - (i % 2) * 16);\n\t            }\n\n\t            return WordArray.create(words, utf16StrLength * 2);\n\t        }\n\t    };\n\n\t    /**\n\t     * UTF-16 LE encoding strategy.\n\t     */\n\t    C_enc.Utf16LE = {\n\t        /**\n\t         * Converts a word array to a UTF-16 LE string.\n\t         *\n\t         * @param {WordArray} wordArray The word array.\n\t         *\n\t         * @return {string} The UTF-16 LE string.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var utf16Str = CryptoJS.enc.Utf16LE.stringify(wordArray);\n\t         */\n\t        stringify: function (wordArray) {\n\t            // Shortcuts\n\t            var words = wordArray.words;\n\t            var sigBytes = wordArray.sigBytes;\n\n\t            // Convert\n\t            var utf16Chars = [];\n\t            for (var i = 0; i < sigBytes; i += 2) {\n\t                var codePoint = swapEndian((words[i >>> 2] >>> (16 - (i % 4) * 8)) & 0xffff);\n\t                utf16Chars.push(String.fromCharCode(codePoint));\n\t            }\n\n\t            return utf16Chars.join('');\n\t        },\n\n\t        /**\n\t         * Converts a UTF-16 LE string to a word array.\n\t         *\n\t         * @param {string} utf16Str The UTF-16 LE string.\n\t         *\n\t         * @return {WordArray} The word array.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.enc.Utf16LE.parse(utf16Str);\n\t         */\n\t        parse: function (utf16Str) {\n\t            // Shortcut\n\t            var utf16StrLength = utf16Str.length;\n\n\t            // Convert\n\t            var words = [];\n\t            for (var i = 0; i < utf16StrLength; i++) {\n\t                words[i >>> 1] |= swapEndian(utf16Str.charCodeAt(i) << (16 - (i % 2) * 16));\n\t            }\n\n\t            return WordArray.create(words, utf16StrLength * 2);\n\t        }\n\t    };\n\n\t    function swapEndian(word) {\n\t        return ((word << 8) & 0xff00ff00) | ((word >>> 8) & 0x00ff00ff);\n\t    }\n\t}());\n\n\n\treturn CryptoJS.enc.Utf16;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./sha1\"), require(\"./hmac\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./sha1\", \"./hmac\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var Base = C_lib.Base;\n\t    var WordArray = C_lib.WordArray;\n\t    var C_algo = C.algo;\n\t    var MD5 = C_algo.MD5;\n\n\t    /**\n\t     * This key derivation function is meant to conform with EVP_BytesToKey.\n\t     * www.openssl.org/docs/crypto/EVP_BytesToKey.html\n\t     */\n\t    var EvpKDF = C_algo.EvpKDF = Base.extend({\n\t        /**\n\t         * Configuration options.\n\t         *\n\t         * @property {number} keySize The key size in words to generate. Default: 4 (128 bits)\n\t         * @property {Hasher} hasher The hash algorithm to use. Default: MD5\n\t         * @property {number} iterations The number of iterations to perform. Default: 1\n\t         */\n\t        cfg: Base.extend({\n\t            keySize: 128/32,\n\t            hasher: MD5,\n\t            iterations: 1\n\t        }),\n\n\t        /**\n\t         * Initializes a newly created key derivation function.\n\t         *\n\t         * @param {Object} cfg (Optional) The configuration options to use for the derivation.\n\t         *\n\t         * @example\n\t         *\n\t         *     var kdf = CryptoJS.algo.EvpKDF.create();\n\t         *     var kdf = CryptoJS.algo.EvpKDF.create({ keySize: 8 });\n\t         *     var kdf = CryptoJS.algo.EvpKDF.create({ keySize: 8, iterations: 1000 });\n\t         */\n\t        init: function (cfg) {\n\t            this.cfg = this.cfg.extend(cfg);\n\t        },\n\n\t        /**\n\t         * Derives a key from a password.\n\t         *\n\t         * @param {WordArray|string} password The password.\n\t         * @param {WordArray|string} salt A salt.\n\t         *\n\t         * @return {WordArray} The derived key.\n\t         *\n\t         * @example\n\t         *\n\t         *     var key = kdf.compute(password, salt);\n\t         */\n\t        compute: function (password, salt) {\n\t            // Shortcut\n\t            var cfg = this.cfg;\n\n\t            // Init hasher\n\t            var hasher = cfg.hasher.create();\n\n\t            // Initial values\n\t            var derivedKey = WordArray.create();\n\n\t            // Shortcuts\n\t            var derivedKeyWords = derivedKey.words;\n\t            var keySize = cfg.keySize;\n\t            var iterations = cfg.iterations;\n\n\t            // Generate key\n\t            while (derivedKeyWords.length < keySize) {\n\t                if (block) {\n\t                    hasher.update(block);\n\t                }\n\t                var block = hasher.update(password).finalize(salt);\n\t                hasher.reset();\n\n\t                // Iterations\n\t                for (var i = 1; i < iterations; i++) {\n\t                    block = hasher.finalize(block);\n\t                    hasher.reset();\n\t                }\n\n\t                derivedKey.concat(block);\n\t            }\n\t            derivedKey.sigBytes = keySize * 4;\n\n\t            return derivedKey;\n\t        }\n\t    });\n\n\t    /**\n\t     * Derives a key from a password.\n\t     *\n\t     * @param {WordArray|string} password The password.\n\t     * @param {WordArray|string} salt A salt.\n\t     * @param {Object} cfg (Optional) The configuration options to use for this computation.\n\t     *\n\t     * @return {WordArray} The derived key.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var key = CryptoJS.EvpKDF(password, salt);\n\t     *     var key = CryptoJS.EvpKDF(password, salt, { keySize: 8 });\n\t     *     var key = CryptoJS.EvpKDF(password, salt, { keySize: 8, iterations: 1000 });\n\t     */\n\t    C.EvpKDF = function (password, salt, cfg) {\n\t        return EvpKDF.create(cfg).compute(password, salt);\n\t    };\n\t}());\n\n\n\treturn CryptoJS.EvpKDF;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function (undefined) {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var CipherParams = C_lib.CipherParams;\n\t    var C_enc = C.enc;\n\t    var Hex = C_enc.Hex;\n\t    var C_format = C.format;\n\n\t    var HexFormatter = C_format.Hex = {\n\t        /**\n\t         * Converts the ciphertext of a cipher params object to a hexadecimally encoded string.\n\t         *\n\t         * @param {CipherParams} cipherParams The cipher params object.\n\t         *\n\t         * @return {string} The hexadecimally encoded string.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var hexString = CryptoJS.format.Hex.stringify(cipherParams);\n\t         */\n\t        stringify: function (cipherParams) {\n\t            return cipherParams.ciphertext.toString(Hex);\n\t        },\n\n\t        /**\n\t         * Converts a hexadecimally encoded ciphertext string to a cipher params object.\n\t         *\n\t         * @param {string} input The hexadecimally encoded string.\n\t         *\n\t         * @return {CipherParams} The cipher params object.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var cipherParams = CryptoJS.format.Hex.parse(hexString);\n\t         */\n\t        parse: function (input) {\n\t            var ciphertext = Hex.parse(input);\n\t            return CipherParams.create({ ciphertext: ciphertext });\n\t        }\n\t    };\n\t}());\n\n\n\treturn CryptoJS.format.Hex;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var Base = C_lib.Base;\n\t    var C_enc = C.enc;\n\t    var Utf8 = C_enc.Utf8;\n\t    var C_algo = C.algo;\n\n\t    /**\n\t     * HMAC algorithm.\n\t     */\n\t    var HMAC = C_algo.HMAC = Base.extend({\n\t        /**\n\t         * Initializes a newly created HMAC.\n\t         *\n\t         * @param {Hasher} hasher The hash algorithm to use.\n\t         * @param {WordArray|string} key The secret key.\n\t         *\n\t         * @example\n\t         *\n\t         *     var hmacHasher = CryptoJS.algo.HMAC.create(CryptoJS.algo.SHA256, key);\n\t         */\n\t        init: function (hasher, key) {\n\t            // Init hasher\n\t            hasher = this._hasher = new hasher.init();\n\n\t            // Convert string to WordArray, else assume WordArray already\n\t            if (typeof key == 'string') {\n\t                key = Utf8.parse(key);\n\t            }\n\n\t            // Shortcuts\n\t            var hasherBlockSize = hasher.blockSize;\n\t            var hasherBlockSizeBytes = hasherBlockSize * 4;\n\n\t            // Allow arbitrary length keys\n\t            if (key.sigBytes > hasherBlockSizeBytes) {\n\t                key = hasher.finalize(key);\n\t            }\n\n\t            // Clamp excess bits\n\t            key.clamp();\n\n\t            // Clone key for inner and outer pads\n\t            var oKey = this._oKey = key.clone();\n\t            var iKey = this._iKey = key.clone();\n\n\t            // Shortcuts\n\t            var oKeyWords = oKey.words;\n\t            var iKeyWords = iKey.words;\n\n\t            // XOR keys with pad constants\n\t            for (var i = 0; i < hasherBlockSize; i++) {\n\t                oKeyWords[i] ^= 0x5c5c5c5c;\n\t                iKeyWords[i] ^= 0x36363636;\n\t            }\n\t            oKey.sigBytes = iKey.sigBytes = hasherBlockSizeBytes;\n\n\t            // Set initial values\n\t            this.reset();\n\t        },\n\n\t        /**\n\t         * Resets this HMAC to its initial state.\n\t         *\n\t         * @example\n\t         *\n\t         *     hmacHasher.reset();\n\t         */\n\t        reset: function () {\n\t            // Shortcut\n\t            var hasher = this._hasher;\n\n\t            // Reset\n\t            hasher.reset();\n\t            hasher.update(this._iKey);\n\t        },\n\n\t        /**\n\t         * Updates this HMAC with a message.\n\t         *\n\t         * @param {WordArray|string} messageUpdate The message to append.\n\t         *\n\t         * @return {HMAC} This HMAC instance.\n\t         *\n\t         * @example\n\t         *\n\t         *     hmacHasher.update('message');\n\t         *     hmacHasher.update(wordArray);\n\t         */\n\t        update: function (messageUpdate) {\n\t            this._hasher.update(messageUpdate);\n\n\t            // Chainable\n\t            return this;\n\t        },\n\n\t        /**\n\t         * Finalizes the HMAC computation.\n\t         * Note that the finalize operation is effectively a destructive, read-once operation.\n\t         *\n\t         * @param {WordArray|string} messageUpdate (Optional) A final message update.\n\t         *\n\t         * @return {WordArray} The HMAC.\n\t         *\n\t         * @example\n\t         *\n\t         *     var hmac = hmacHasher.finalize();\n\t         *     var hmac = hmacHasher.finalize('message');\n\t         *     var hmac = hmacHasher.finalize(wordArray);\n\t         */\n\t        finalize: function (messageUpdate) {\n\t            // Shortcut\n\t            var hasher = this._hasher;\n\n\t            // Compute HMAC\n\t            var innerHash = hasher.finalize(messageUpdate);\n\t            hasher.reset();\n\t            var hmac = hasher.finalize(this._oKey.clone().concat(innerHash));\n\n\t            return hmac;\n\t        }\n\t    });\n\t}());\n\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./x64-core\"), require(\"./lib-typedarrays\"), require(\"./enc-utf16\"), require(\"./enc-base64\"), require(\"./md5\"), require(\"./sha1\"), require(\"./sha256\"), require(\"./sha224\"), require(\"./sha512\"), require(\"./sha384\"), require(\"./sha3\"), require(\"./ripemd160\"), require(\"./hmac\"), require(\"./pbkdf2\"), require(\"./evpkdf\"), require(\"./cipher-core\"), require(\"./mode-cfb\"), require(\"./mode-ctr\"), require(\"./mode-ctr-gladman\"), require(\"./mode-ofb\"), require(\"./mode-ecb\"), require(\"./pad-ansix923\"), require(\"./pad-iso10126\"), require(\"./pad-iso97971\"), require(\"./pad-zeropadding\"), require(\"./pad-nopadding\"), require(\"./format-hex\"), require(\"./aes\"), require(\"./tripledes\"), require(\"./rc4\"), require(\"./rabbit\"), require(\"./rabbit-legacy\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./x64-core\", \"./lib-typedarrays\", \"./enc-utf16\", \"./enc-base64\", \"./md5\", \"./sha1\", \"./sha256\", \"./sha224\", \"./sha512\", \"./sha384\", \"./sha3\", \"./ripemd160\", \"./hmac\", \"./pbkdf2\", \"./evpkdf\", \"./cipher-core\", \"./mode-cfb\", \"./mode-ctr\", \"./mode-ctr-gladman\", \"./mode-ofb\", \"./mode-ecb\", \"./pad-ansix923\", \"./pad-iso10126\", \"./pad-iso97971\", \"./pad-zeropadding\", \"./pad-nopadding\", \"./format-hex\", \"./aes\", \"./tripledes\", \"./rc4\", \"./rabbit\", \"./rabbit-legacy\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\troot.CryptoJS = factory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\treturn CryptoJS;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Check if typed arrays are supported\n\t    if (typeof ArrayBuffer != 'function') {\n\t        return;\n\t    }\n\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\n\t    // Reference original init\n\t    var superInit = WordArray.init;\n\n\t    // Augment WordArray.init to handle typed arrays\n\t    var subInit = WordArray.init = function (typedArray) {\n\t        // Convert buffers to uint8\n\t        if (typedArray instanceof ArrayBuffer) {\n\t            typedArray = new Uint8Array(typedArray);\n\t        }\n\n\t        // Convert other array views to uint8\n\t        if (\n\t            typedArray instanceof Int8Array ||\n\t            (typeof Uint8ClampedArray !== \"undefined\" && typedArray instanceof Uint8ClampedArray) ||\n\t            typedArray instanceof Int16Array ||\n\t            typedArray instanceof Uint16Array ||\n\t            typedArray instanceof Int32Array ||\n\t            typedArray instanceof Uint32Array ||\n\t            typedArray instanceof Float32Array ||\n\t            typedArray instanceof Float64Array\n\t        ) {\n\t            typedArray = new Uint8Array(typedArray.buffer, typedArray.byteOffset, typedArray.byteLength);\n\t        }\n\n\t        // Handle Uint8Array\n\t        if (typedArray instanceof Uint8Array) {\n\t            // Shortcut\n\t            var typedArrayByteLength = typedArray.byteLength;\n\n\t            // Extract bytes\n\t            var words = [];\n\t            for (var i = 0; i < typedArrayByteLength; i++) {\n\t                words[i >>> 2] |= typedArray[i] << (24 - (i % 4) * 8);\n\t            }\n\n\t            // Initialize this word array\n\t            superInit.call(this, words, typedArrayByteLength);\n\t        } else {\n\t            // Else call normal init\n\t            superInit.apply(this, arguments);\n\t        }\n\t    };\n\n\t    subInit.prototype = WordArray;\n\t}());\n\n\n\treturn CryptoJS.lib.WordArray;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function (Math) {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var Hasher = C_lib.Hasher;\n\t    var C_algo = C.algo;\n\n\t    // Constants table\n\t    var T = [];\n\n\t    // Compute constants\n\t    (function () {\n\t        for (var i = 0; i < 64; i++) {\n\t            T[i] = (Math.abs(Math.sin(i + 1)) * 0x100000000) | 0;\n\t        }\n\t    }());\n\n\t    /**\n\t     * MD5 hash algorithm.\n\t     */\n\t    var MD5 = C_algo.MD5 = Hasher.extend({\n\t        _doReset: function () {\n\t            this._hash = new WordArray.init([\n\t                0x67452301, 0xefcdab89,\n\t                0x98badcfe, 0x10325476\n\t            ]);\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\t            // Swap endian\n\t            for (var i = 0; i < 16; i++) {\n\t                // Shortcuts\n\t                var offset_i = offset + i;\n\t                var M_offset_i = M[offset_i];\n\n\t                M[offset_i] = (\n\t                    (((M_offset_i << 8)  | (M_offset_i >>> 24)) & 0x00ff00ff) |\n\t                    (((M_offset_i << 24) | (M_offset_i >>> 8))  & 0xff00ff00)\n\t                );\n\t            }\n\n\t            // Shortcuts\n\t            var H = this._hash.words;\n\n\t            var M_offset_0  = M[offset + 0];\n\t            var M_offset_1  = M[offset + 1];\n\t            var M_offset_2  = M[offset + 2];\n\t            var M_offset_3  = M[offset + 3];\n\t            var M_offset_4  = M[offset + 4];\n\t            var M_offset_5  = M[offset + 5];\n\t            var M_offset_6  = M[offset + 6];\n\t            var M_offset_7  = M[offset + 7];\n\t            var M_offset_8  = M[offset + 8];\n\t            var M_offset_9  = M[offset + 9];\n\t            var M_offset_10 = M[offset + 10];\n\t            var M_offset_11 = M[offset + 11];\n\t            var M_offset_12 = M[offset + 12];\n\t            var M_offset_13 = M[offset + 13];\n\t            var M_offset_14 = M[offset + 14];\n\t            var M_offset_15 = M[offset + 15];\n\n\t            // Working varialbes\n\t            var a = H[0];\n\t            var b = H[1];\n\t            var c = H[2];\n\t            var d = H[3];\n\n\t            // Computation\n\t            a = FF(a, b, c, d, M_offset_0,  7,  T[0]);\n\t            d = FF(d, a, b, c, M_offset_1,  12, T[1]);\n\t            c = FF(c, d, a, b, M_offset_2,  17, T[2]);\n\t            b = FF(b, c, d, a, M_offset_3,  22, T[3]);\n\t            a = FF(a, b, c, d, M_offset_4,  7,  T[4]);\n\t            d = FF(d, a, b, c, M_offset_5,  12, T[5]);\n\t            c = FF(c, d, a, b, M_offset_6,  17, T[6]);\n\t            b = FF(b, c, d, a, M_offset_7,  22, T[7]);\n\t            a = FF(a, b, c, d, M_offset_8,  7,  T[8]);\n\t            d = FF(d, a, b, c, M_offset_9,  12, T[9]);\n\t            c = FF(c, d, a, b, M_offset_10, 17, T[10]);\n\t            b = FF(b, c, d, a, M_offset_11, 22, T[11]);\n\t            a = FF(a, b, c, d, M_offset_12, 7,  T[12]);\n\t            d = FF(d, a, b, c, M_offset_13, 12, T[13]);\n\t            c = FF(c, d, a, b, M_offset_14, 17, T[14]);\n\t            b = FF(b, c, d, a, M_offset_15, 22, T[15]);\n\n\t            a = GG(a, b, c, d, M_offset_1,  5,  T[16]);\n\t            d = GG(d, a, b, c, M_offset_6,  9,  T[17]);\n\t            c = GG(c, d, a, b, M_offset_11, 14, T[18]);\n\t            b = GG(b, c, d, a, M_offset_0,  20, T[19]);\n\t            a = GG(a, b, c, d, M_offset_5,  5,  T[20]);\n\t            d = GG(d, a, b, c, M_offset_10, 9,  T[21]);\n\t            c = GG(c, d, a, b, M_offset_15, 14, T[22]);\n\t            b = GG(b, c, d, a, M_offset_4,  20, T[23]);\n\t            a = GG(a, b, c, d, M_offset_9,  5,  T[24]);\n\t            d = GG(d, a, b, c, M_offset_14, 9,  T[25]);\n\t            c = GG(c, d, a, b, M_offset_3,  14, T[26]);\n\t            b = GG(b, c, d, a, M_offset_8,  20, T[27]);\n\t            a = GG(a, b, c, d, M_offset_13, 5,  T[28]);\n\t            d = GG(d, a, b, c, M_offset_2,  9,  T[29]);\n\t            c = GG(c, d, a, b, M_offset_7,  14, T[30]);\n\t            b = GG(b, c, d, a, M_offset_12, 20, T[31]);\n\n\t            a = HH(a, b, c, d, M_offset_5,  4,  T[32]);\n\t            d = HH(d, a, b, c, M_offset_8,  11, T[33]);\n\t            c = HH(c, d, a, b, M_offset_11, 16, T[34]);\n\t            b = HH(b, c, d, a, M_offset_14, 23, T[35]);\n\t            a = HH(a, b, c, d, M_offset_1,  4,  T[36]);\n\t            d = HH(d, a, b, c, M_offset_4,  11, T[37]);\n\t            c = HH(c, d, a, b, M_offset_7,  16, T[38]);\n\t            b = HH(b, c, d, a, M_offset_10, 23, T[39]);\n\t            a = HH(a, b, c, d, M_offset_13, 4,  T[40]);\n\t            d = HH(d, a, b, c, M_offset_0,  11, T[41]);\n\t            c = HH(c, d, a, b, M_offset_3,  16, T[42]);\n\t            b = HH(b, c, d, a, M_offset_6,  23, T[43]);\n\t            a = HH(a, b, c, d, M_offset_9,  4,  T[44]);\n\t            d = HH(d, a, b, c, M_offset_12, 11, T[45]);\n\t            c = HH(c, d, a, b, M_offset_15, 16, T[46]);\n\t            b = HH(b, c, d, a, M_offset_2,  23, T[47]);\n\n\t            a = II(a, b, c, d, M_offset_0,  6,  T[48]);\n\t            d = II(d, a, b, c, M_offset_7,  10, T[49]);\n\t            c = II(c, d, a, b, M_offset_14, 15, T[50]);\n\t            b = II(b, c, d, a, M_offset_5,  21, T[51]);\n\t            a = II(a, b, c, d, M_offset_12, 6,  T[52]);\n\t            d = II(d, a, b, c, M_offset_3,  10, T[53]);\n\t            c = II(c, d, a, b, M_offset_10, 15, T[54]);\n\t            b = II(b, c, d, a, M_offset_1,  21, T[55]);\n\t            a = II(a, b, c, d, M_offset_8,  6,  T[56]);\n\t            d = II(d, a, b, c, M_offset_15, 10, T[57]);\n\t            c = II(c, d, a, b, M_offset_6,  15, T[58]);\n\t            b = II(b, c, d, a, M_offset_13, 21, T[59]);\n\t            a = II(a, b, c, d, M_offset_4,  6,  T[60]);\n\t            d = II(d, a, b, c, M_offset_11, 10, T[61]);\n\t            c = II(c, d, a, b, M_offset_2,  15, T[62]);\n\t            b = II(b, c, d, a, M_offset_9,  21, T[63]);\n\n\t            // Intermediate hash value\n\t            H[0] = (H[0] + a) | 0;\n\t            H[1] = (H[1] + b) | 0;\n\t            H[2] = (H[2] + c) | 0;\n\t            H[3] = (H[3] + d) | 0;\n\t        },\n\n\t        _doFinalize: function () {\n\t            // Shortcuts\n\t            var data = this._data;\n\t            var dataWords = data.words;\n\n\t            var nBitsTotal = this._nDataBytes * 8;\n\t            var nBitsLeft = data.sigBytes * 8;\n\n\t            // Add padding\n\t            dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);\n\n\t            var nBitsTotalH = Math.floor(nBitsTotal / 0x100000000);\n\t            var nBitsTotalL = nBitsTotal;\n\t            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 15] = (\n\t                (((nBitsTotalH << 8)  | (nBitsTotalH >>> 24)) & 0x00ff00ff) |\n\t                (((nBitsTotalH << 24) | (nBitsTotalH >>> 8))  & 0xff00ff00)\n\t            );\n\t            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = (\n\t                (((nBitsTotalL << 8)  | (nBitsTotalL >>> 24)) & 0x00ff00ff) |\n\t                (((nBitsTotalL << 24) | (nBitsTotalL >>> 8))  & 0xff00ff00)\n\t            );\n\n\t            data.sigBytes = (dataWords.length + 1) * 4;\n\n\t            // Hash final blocks\n\t            this._process();\n\n\t            // Shortcuts\n\t            var hash = this._hash;\n\t            var H = hash.words;\n\n\t            // Swap endian\n\t            for (var i = 0; i < 4; i++) {\n\t                // Shortcut\n\t                var H_i = H[i];\n\n\t                H[i] = (((H_i << 8)  | (H_i >>> 24)) & 0x00ff00ff) |\n\t                       (((H_i << 24) | (H_i >>> 8))  & 0xff00ff00);\n\t            }\n\n\t            // Return final computed hash\n\t            return hash;\n\t        },\n\n\t        clone: function () {\n\t            var clone = Hasher.clone.call(this);\n\t            clone._hash = this._hash.clone();\n\n\t            return clone;\n\t        }\n\t    });\n\n\t    function FF(a, b, c, d, x, s, t) {\n\t        var n = a + ((b & c) | (~b & d)) + x + t;\n\t        return ((n << s) | (n >>> (32 - s))) + b;\n\t    }\n\n\t    function GG(a, b, c, d, x, s, t) {\n\t        var n = a + ((b & d) | (c & ~d)) + x + t;\n\t        return ((n << s) | (n >>> (32 - s))) + b;\n\t    }\n\n\t    function HH(a, b, c, d, x, s, t) {\n\t        var n = a + (b ^ c ^ d) + x + t;\n\t        return ((n << s) | (n >>> (32 - s))) + b;\n\t    }\n\n\t    function II(a, b, c, d, x, s, t) {\n\t        var n = a + (c ^ (b | ~d)) + x + t;\n\t        return ((n << s) | (n >>> (32 - s))) + b;\n\t    }\n\n\t    /**\n\t     * Shortcut function to the hasher's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     *\n\t     * @return {WordArray} The hash.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hash = CryptoJS.MD5('message');\n\t     *     var hash = CryptoJS.MD5(wordArray);\n\t     */\n\t    C.MD5 = Hasher._createHelper(MD5);\n\n\t    /**\n\t     * Shortcut function to the HMAC's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     * @param {WordArray|string} key The secret key.\n\t     *\n\t     * @return {WordArray} The HMAC.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hmac = CryptoJS.HmacMD5(message, key);\n\t     */\n\t    C.HmacMD5 = Hasher._createHmacHelper(MD5);\n\t}(Math));\n\n\n\treturn CryptoJS.MD5;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * Cipher Feedback block mode.\n\t */\n\tCryptoJS.mode.CFB = (function () {\n\t    var CFB = CryptoJS.lib.BlockCipherMode.extend();\n\n\t    CFB.Encryptor = CFB.extend({\n\t        processBlock: function (words, offset) {\n\t            // Shortcuts\n\t            var cipher = this._cipher;\n\t            var blockSize = cipher.blockSize;\n\n\t            generateKeystreamAndEncrypt.call(this, words, offset, blockSize, cipher);\n\n\t            // Remember this block to use with next block\n\t            this._prevBlock = words.slice(offset, offset + blockSize);\n\t        }\n\t    });\n\n\t    CFB.Decryptor = CFB.extend({\n\t        processBlock: function (words, offset) {\n\t            // Shortcuts\n\t            var cipher = this._cipher;\n\t            var blockSize = cipher.blockSize;\n\n\t            // Remember this block to use with next block\n\t            var thisBlock = words.slice(offset, offset + blockSize);\n\n\t            generateKeystreamAndEncrypt.call(this, words, offset, blockSize, cipher);\n\n\t            // This block becomes the previous block\n\t            this._prevBlock = thisBlock;\n\t        }\n\t    });\n\n\t    function generateKeystreamAndEncrypt(words, offset, blockSize, cipher) {\n\t        // Shortcut\n\t        var iv = this._iv;\n\n\t        // Generate keystream\n\t        if (iv) {\n\t            var keystream = iv.slice(0);\n\n\t            // Remove IV for subsequent blocks\n\t            this._iv = undefined;\n\t        } else {\n\t            var keystream = this._prevBlock;\n\t        }\n\t        cipher.encryptBlock(keystream, 0);\n\n\t        // Encrypt\n\t        for (var i = 0; i < blockSize; i++) {\n\t            words[offset + i] ^= keystream[i];\n\t        }\n\t    }\n\n\t    return CFB;\n\t}());\n\n\n\treturn CryptoJS.mode.CFB;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/** @preserve\n\t * Counter block mode compatible with  Dr Brian Gladman fileenc.c\n\t * derived from CryptoJS.mode.CTR\n\t * Jan Hruby jhruby.web@gmail.com\n\t */\n\tCryptoJS.mode.CTRGladman = (function () {\n\t    var CTRGladman = CryptoJS.lib.BlockCipherMode.extend();\n\n\t\tfunction incWord(word)\n\t\t{\n\t\t\tif (((word >> 24) & 0xff) === 0xff) { //overflow\n\t\t\tvar b1 = (word >> 16)&0xff;\n\t\t\tvar b2 = (word >> 8)&0xff;\n\t\t\tvar b3 = word & 0xff;\n\n\t\t\tif (b1 === 0xff) // overflow b1\n\t\t\t{\n\t\t\tb1 = 0;\n\t\t\tif (b2 === 0xff)\n\t\t\t{\n\t\t\t\tb2 = 0;\n\t\t\t\tif (b3 === 0xff)\n\t\t\t\t{\n\t\t\t\t\tb3 = 0;\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\t++b3;\n\t\t\t\t}\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t++b2;\n\t\t\t}\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t++b1;\n\t\t\t}\n\n\t\t\tword = 0;\n\t\t\tword += (b1 << 16);\n\t\t\tword += (b2 << 8);\n\t\t\tword += b3;\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\tword += (0x01 << 24);\n\t\t\t}\n\t\t\treturn word;\n\t\t}\n\n\t\tfunction incCounter(counter)\n\t\t{\n\t\t\tif ((counter[0] = incWord(counter[0])) === 0)\n\t\t\t{\n\t\t\t\t// encr_data in fileenc.c from  Dr Brian Gladman's counts only with DWORD j < 8\n\t\t\t\tcounter[1] = incWord(counter[1]);\n\t\t\t}\n\t\t\treturn counter;\n\t\t}\n\n\t    var Encryptor = CTRGladman.Encryptor = CTRGladman.extend({\n\t        processBlock: function (words, offset) {\n\t            // Shortcuts\n\t            var cipher = this._cipher\n\t            var blockSize = cipher.blockSize;\n\t            var iv = this._iv;\n\t            var counter = this._counter;\n\n\t            // Generate keystream\n\t            if (iv) {\n\t                counter = this._counter = iv.slice(0);\n\n\t                // Remove IV for subsequent blocks\n\t                this._iv = undefined;\n\t            }\n\n\t\t\t\tincCounter(counter);\n\n\t\t\t\tvar keystream = counter.slice(0);\n\t            cipher.encryptBlock(keystream, 0);\n\n\t            // Encrypt\n\t            for (var i = 0; i < blockSize; i++) {\n\t                words[offset + i] ^= keystream[i];\n\t            }\n\t        }\n\t    });\n\n\t    CTRGladman.Decryptor = Encryptor;\n\n\t    return CTRGladman;\n\t}());\n\n\n\n\n\treturn CryptoJS.mode.CTRGladman;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * Counter block mode.\n\t */\n\tCryptoJS.mode.CTR = (function () {\n\t    var CTR = CryptoJS.lib.BlockCipherMode.extend();\n\n\t    var Encryptor = CTR.Encryptor = CTR.extend({\n\t        processBlock: function (words, offset) {\n\t            // Shortcuts\n\t            var cipher = this._cipher\n\t            var blockSize = cipher.blockSize;\n\t            var iv = this._iv;\n\t            var counter = this._counter;\n\n\t            // Generate keystream\n\t            if (iv) {\n\t                counter = this._counter = iv.slice(0);\n\n\t                // Remove IV for subsequent blocks\n\t                this._iv = undefined;\n\t            }\n\t            var keystream = counter.slice(0);\n\t            cipher.encryptBlock(keystream, 0);\n\n\t            // Increment counter\n\t            counter[blockSize - 1] = (counter[blockSize - 1] + 1) | 0\n\n\t            // Encrypt\n\t            for (var i = 0; i < blockSize; i++) {\n\t                words[offset + i] ^= keystream[i];\n\t            }\n\t        }\n\t    });\n\n\t    CTR.Decryptor = Encryptor;\n\n\t    return CTR;\n\t}());\n\n\n\treturn CryptoJS.mode.CTR;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * Electronic Codebook block mode.\n\t */\n\tCryptoJS.mode.ECB = (function () {\n\t    var ECB = CryptoJS.lib.BlockCipherMode.extend();\n\n\t    ECB.Encryptor = ECB.extend({\n\t        processBlock: function (words, offset) {\n\t            this._cipher.encryptBlock(words, offset);\n\t        }\n\t    });\n\n\t    ECB.Decryptor = ECB.extend({\n\t        processBlock: function (words, offset) {\n\t            this._cipher.decryptBlock(words, offset);\n\t        }\n\t    });\n\n\t    return ECB;\n\t}());\n\n\n\treturn CryptoJS.mode.ECB;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * Output Feedback block mode.\n\t */\n\tCryptoJS.mode.OFB = (function () {\n\t    var OFB = CryptoJS.lib.BlockCipherMode.extend();\n\n\t    var Encryptor = OFB.Encryptor = OFB.extend({\n\t        processBlock: function (words, offset) {\n\t            // Shortcuts\n\t            var cipher = this._cipher\n\t            var blockSize = cipher.blockSize;\n\t            var iv = this._iv;\n\t            var keystream = this._keystream;\n\n\t            // Generate keystream\n\t            if (iv) {\n\t                keystream = this._keystream = iv.slice(0);\n\n\t                // Remove IV for subsequent blocks\n\t                this._iv = undefined;\n\t            }\n\t            cipher.encryptBlock(keystream, 0);\n\n\t            // Encrypt\n\t            for (var i = 0; i < blockSize; i++) {\n\t                words[offset + i] ^= keystream[i];\n\t            }\n\t        }\n\t    });\n\n\t    OFB.Decryptor = Encryptor;\n\n\t    return OFB;\n\t}());\n\n\n\treturn CryptoJS.mode.OFB;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * ANSI X.923 padding strategy.\n\t */\n\tCryptoJS.pad.AnsiX923 = {\n\t    pad: function (data, blockSize) {\n\t        // Shortcuts\n\t        var dataSigBytes = data.sigBytes;\n\t        var blockSizeBytes = blockSize * 4;\n\n\t        // Count padding bytes\n\t        var nPaddingBytes = blockSizeBytes - dataSigBytes % blockSizeBytes;\n\n\t        // Compute last byte position\n\t        var lastBytePos = dataSigBytes + nPaddingBytes - 1;\n\n\t        // Pad\n\t        data.clamp();\n\t        data.words[lastBytePos >>> 2] |= nPaddingBytes << (24 - (lastBytePos % 4) * 8);\n\t        data.sigBytes += nPaddingBytes;\n\t    },\n\n\t    unpad: function (data) {\n\t        // Get number of padding bytes from last byte\n\t        var nPaddingBytes = data.words[(data.sigBytes - 1) >>> 2] & 0xff;\n\n\t        // Remove padding\n\t        data.sigBytes -= nPaddingBytes;\n\t    }\n\t};\n\n\n\treturn CryptoJS.pad.Ansix923;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * ISO 10126 padding strategy.\n\t */\n\tCryptoJS.pad.Iso10126 = {\n\t    pad: function (data, blockSize) {\n\t        // Shortcut\n\t        var blockSizeBytes = blockSize * 4;\n\n\t        // Count padding bytes\n\t        var nPaddingBytes = blockSizeBytes - data.sigBytes % blockSizeBytes;\n\n\t        // Pad\n\t        data.concat(CryptoJS.lib.WordArray.random(nPaddingBytes - 1)).\n\t             concat(CryptoJS.lib.WordArray.create([nPaddingBytes << 24], 1));\n\t    },\n\n\t    unpad: function (data) {\n\t        // Get number of padding bytes from last byte\n\t        var nPaddingBytes = data.words[(data.sigBytes - 1) >>> 2] & 0xff;\n\n\t        // Remove padding\n\t        data.sigBytes -= nPaddingBytes;\n\t    }\n\t};\n\n\n\treturn CryptoJS.pad.Iso10126;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * ISO/IEC 9797-1 Padding Method 2.\n\t */\n\tCryptoJS.pad.Iso97971 = {\n\t    pad: function (data, blockSize) {\n\t        // Add 0x80 byte\n\t        data.concat(CryptoJS.lib.WordArray.create([0x80000000], 1));\n\n\t        // Zero pad the rest\n\t        CryptoJS.pad.ZeroPadding.pad(data, blockSize);\n\t    },\n\n\t    unpad: function (data) {\n\t        // Remove zero padding\n\t        CryptoJS.pad.ZeroPadding.unpad(data);\n\n\t        // Remove one more byte -- the 0x80 byte\n\t        data.sigBytes--;\n\t    }\n\t};\n\n\n\treturn CryptoJS.pad.Iso97971;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * A noop padding strategy.\n\t */\n\tCryptoJS.pad.NoPadding = {\n\t    pad: function () {\n\t    },\n\n\t    unpad: function () {\n\t    }\n\t};\n\n\n\treturn CryptoJS.pad.NoPadding;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * Zero padding strategy.\n\t */\n\tCryptoJS.pad.ZeroPadding = {\n\t    pad: function (data, blockSize) {\n\t        // Shortcut\n\t        var blockSizeBytes = blockSize * 4;\n\n\t        // Pad\n\t        data.clamp();\n\t        data.sigBytes += blockSizeBytes - ((data.sigBytes % blockSizeBytes) || blockSizeBytes);\n\t    },\n\n\t    unpad: function (data) {\n\t        // Shortcut\n\t        var dataWords = data.words;\n\n\t        // Unpad\n\t        var i = data.sigBytes - 1;\n\t        while (!((dataWords[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff)) {\n\t            i--;\n\t        }\n\t        data.sigBytes = i + 1;\n\t    }\n\t};\n\n\n\treturn CryptoJS.pad.ZeroPadding;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./sha1\"), require(\"./hmac\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./sha1\", \"./hmac\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var Base = C_lib.Base;\n\t    var WordArray = C_lib.WordArray;\n\t    var C_algo = C.algo;\n\t    var SHA1 = C_algo.SHA1;\n\t    var HMAC = C_algo.HMAC;\n\n\t    /**\n\t     * Password-Based Key Derivation Function 2 algorithm.\n\t     */\n\t    var PBKDF2 = C_algo.PBKDF2 = Base.extend({\n\t        /**\n\t         * Configuration options.\n\t         *\n\t         * @property {number} keySize The key size in words to generate. Default: 4 (128 bits)\n\t         * @property {Hasher} hasher The hasher to use. Default: SHA1\n\t         * @property {number} iterations The number of iterations to perform. Default: 1\n\t         */\n\t        cfg: Base.extend({\n\t            keySize: 128/32,\n\t            hasher: SHA1,\n\t            iterations: 1\n\t        }),\n\n\t        /**\n\t         * Initializes a newly created key derivation function.\n\t         *\n\t         * @param {Object} cfg (Optional) The configuration options to use for the derivation.\n\t         *\n\t         * @example\n\t         *\n\t         *     var kdf = CryptoJS.algo.PBKDF2.create();\n\t         *     var kdf = CryptoJS.algo.PBKDF2.create({ keySize: 8 });\n\t         *     var kdf = CryptoJS.algo.PBKDF2.create({ keySize: 8, iterations: 1000 });\n\t         */\n\t        init: function (cfg) {\n\t            this.cfg = this.cfg.extend(cfg);\n\t        },\n\n\t        /**\n\t         * Computes the Password-Based Key Derivation Function 2.\n\t         *\n\t         * @param {WordArray|string} password The password.\n\t         * @param {WordArray|string} salt A salt.\n\t         *\n\t         * @return {WordArray} The derived key.\n\t         *\n\t         * @example\n\t         *\n\t         *     var key = kdf.compute(password, salt);\n\t         */\n\t        compute: function (password, salt) {\n\t            // Shortcut\n\t            var cfg = this.cfg;\n\n\t            // Init HMAC\n\t            var hmac = HMAC.create(cfg.hasher, password);\n\n\t            // Initial values\n\t            var derivedKey = WordArray.create();\n\t            var blockIndex = WordArray.create([0x00000001]);\n\n\t            // Shortcuts\n\t            var derivedKeyWords = derivedKey.words;\n\t            var blockIndexWords = blockIndex.words;\n\t            var keySize = cfg.keySize;\n\t            var iterations = cfg.iterations;\n\n\t            // Generate key\n\t            while (derivedKeyWords.length < keySize) {\n\t                var block = hmac.update(salt).finalize(blockIndex);\n\t                hmac.reset();\n\n\t                // Shortcuts\n\t                var blockWords = block.words;\n\t                var blockWordsLength = blockWords.length;\n\n\t                // Iterations\n\t                var intermediate = block;\n\t                for (var i = 1; i < iterations; i++) {\n\t                    intermediate = hmac.finalize(intermediate);\n\t                    hmac.reset();\n\n\t                    // Shortcut\n\t                    var intermediateWords = intermediate.words;\n\n\t                    // XOR intermediate with block\n\t                    for (var j = 0; j < blockWordsLength; j++) {\n\t                        blockWords[j] ^= intermediateWords[j];\n\t                    }\n\t                }\n\n\t                derivedKey.concat(block);\n\t                blockIndexWords[0]++;\n\t            }\n\t            derivedKey.sigBytes = keySize * 4;\n\n\t            return derivedKey;\n\t        }\n\t    });\n\n\t    /**\n\t     * Computes the Password-Based Key Derivation Function 2.\n\t     *\n\t     * @param {WordArray|string} password The password.\n\t     * @param {WordArray|string} salt A salt.\n\t     * @param {Object} cfg (Optional) The configuration options to use for this computation.\n\t     *\n\t     * @return {WordArray} The derived key.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var key = CryptoJS.PBKDF2(password, salt);\n\t     *     var key = CryptoJS.PBKDF2(password, salt, { keySize: 8 });\n\t     *     var key = CryptoJS.PBKDF2(password, salt, { keySize: 8, iterations: 1000 });\n\t     */\n\t    C.PBKDF2 = function (password, salt, cfg) {\n\t        return PBKDF2.create(cfg).compute(password, salt);\n\t    };\n\t}());\n\n\n\treturn CryptoJS.PBKDF2;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./enc-base64\"), require(\"./md5\"), require(\"./evpkdf\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./enc-base64\", \"./md5\", \"./evpkdf\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var StreamCipher = C_lib.StreamCipher;\n\t    var C_algo = C.algo;\n\n\t    // Reusable objects\n\t    var S  = [];\n\t    var C_ = [];\n\t    var G  = [];\n\n\t    /**\n\t     * Rabbit stream cipher algorithm.\n\t     *\n\t     * This is a legacy version that neglected to convert the key to little-endian.\n\t     * This error doesn't affect the cipher's security,\n\t     * but it does affect its compatibility with other implementations.\n\t     */\n\t    var RabbitLegacy = C_algo.RabbitLegacy = StreamCipher.extend({\n\t        _doReset: function () {\n\t            // Shortcuts\n\t            var K = this._key.words;\n\t            var iv = this.cfg.iv;\n\n\t            // Generate initial state values\n\t            var X = this._X = [\n\t                K[0], (K[3] << 16) | (K[2] >>> 16),\n\t                K[1], (K[0] << 16) | (K[3] >>> 16),\n\t                K[2], (K[1] << 16) | (K[0] >>> 16),\n\t                K[3], (K[2] << 16) | (K[1] >>> 16)\n\t            ];\n\n\t            // Generate initial counter values\n\t            var C = this._C = [\n\t                (K[2] << 16) | (K[2] >>> 16), (K[0] & 0xffff0000) | (K[1] & 0x0000ffff),\n\t                (K[3] << 16) | (K[3] >>> 16), (K[1] & 0xffff0000) | (K[2] & 0x0000ffff),\n\t                (K[0] << 16) | (K[0] >>> 16), (K[2] & 0xffff0000) | (K[3] & 0x0000ffff),\n\t                (K[1] << 16) | (K[1] >>> 16), (K[3] & 0xffff0000) | (K[0] & 0x0000ffff)\n\t            ];\n\n\t            // Carry bit\n\t            this._b = 0;\n\n\t            // Iterate the system four times\n\t            for (var i = 0; i < 4; i++) {\n\t                nextState.call(this);\n\t            }\n\n\t            // Modify the counters\n\t            for (var i = 0; i < 8; i++) {\n\t                C[i] ^= X[(i + 4) & 7];\n\t            }\n\n\t            // IV setup\n\t            if (iv) {\n\t                // Shortcuts\n\t                var IV = iv.words;\n\t                var IV_0 = IV[0];\n\t                var IV_1 = IV[1];\n\n\t                // Generate four subvectors\n\t                var i0 = (((IV_0 << 8) | (IV_0 >>> 24)) & 0x00ff00ff) | (((IV_0 << 24) | (IV_0 >>> 8)) & 0xff00ff00);\n\t                var i2 = (((IV_1 << 8) | (IV_1 >>> 24)) & 0x00ff00ff) | (((IV_1 << 24) | (IV_1 >>> 8)) & 0xff00ff00);\n\t                var i1 = (i0 >>> 16) | (i2 & 0xffff0000);\n\t                var i3 = (i2 << 16)  | (i0 & 0x0000ffff);\n\n\t                // Modify counter values\n\t                C[0] ^= i0;\n\t                C[1] ^= i1;\n\t                C[2] ^= i2;\n\t                C[3] ^= i3;\n\t                C[4] ^= i0;\n\t                C[5] ^= i1;\n\t                C[6] ^= i2;\n\t                C[7] ^= i3;\n\n\t                // Iterate the system four times\n\t                for (var i = 0; i < 4; i++) {\n\t                    nextState.call(this);\n\t                }\n\t            }\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\t            // Shortcut\n\t            var X = this._X;\n\n\t            // Iterate the system\n\t            nextState.call(this);\n\n\t            // Generate four keystream words\n\t            S[0] = X[0] ^ (X[5] >>> 16) ^ (X[3] << 16);\n\t            S[1] = X[2] ^ (X[7] >>> 16) ^ (X[5] << 16);\n\t            S[2] = X[4] ^ (X[1] >>> 16) ^ (X[7] << 16);\n\t            S[3] = X[6] ^ (X[3] >>> 16) ^ (X[1] << 16);\n\n\t            for (var i = 0; i < 4; i++) {\n\t                // Swap endian\n\t                S[i] = (((S[i] << 8)  | (S[i] >>> 24)) & 0x00ff00ff) |\n\t                       (((S[i] << 24) | (S[i] >>> 8))  & 0xff00ff00);\n\n\t                // Encrypt\n\t                M[offset + i] ^= S[i];\n\t            }\n\t        },\n\n\t        blockSize: 128/32,\n\n\t        ivSize: 64/32\n\t    });\n\n\t    function nextState() {\n\t        // Shortcuts\n\t        var X = this._X;\n\t        var C = this._C;\n\n\t        // Save old counter values\n\t        for (var i = 0; i < 8; i++) {\n\t            C_[i] = C[i];\n\t        }\n\n\t        // Calculate new counter values\n\t        C[0] = (C[0] + 0x4d34d34d + this._b) | 0;\n\t        C[1] = (C[1] + 0xd34d34d3 + ((C[0] >>> 0) < (C_[0] >>> 0) ? 1 : 0)) | 0;\n\t        C[2] = (C[2] + 0x34d34d34 + ((C[1] >>> 0) < (C_[1] >>> 0) ? 1 : 0)) | 0;\n\t        C[3] = (C[3] + 0x4d34d34d + ((C[2] >>> 0) < (C_[2] >>> 0) ? 1 : 0)) | 0;\n\t        C[4] = (C[4] + 0xd34d34d3 + ((C[3] >>> 0) < (C_[3] >>> 0) ? 1 : 0)) | 0;\n\t        C[5] = (C[5] + 0x34d34d34 + ((C[4] >>> 0) < (C_[4] >>> 0) ? 1 : 0)) | 0;\n\t        C[6] = (C[6] + 0x4d34d34d + ((C[5] >>> 0) < (C_[5] >>> 0) ? 1 : 0)) | 0;\n\t        C[7] = (C[7] + 0xd34d34d3 + ((C[6] >>> 0) < (C_[6] >>> 0) ? 1 : 0)) | 0;\n\t        this._b = (C[7] >>> 0) < (C_[7] >>> 0) ? 1 : 0;\n\n\t        // Calculate the g-values\n\t        for (var i = 0; i < 8; i++) {\n\t            var gx = X[i] + C[i];\n\n\t            // Construct high and low argument for squaring\n\t            var ga = gx & 0xffff;\n\t            var gb = gx >>> 16;\n\n\t            // Calculate high and low result of squaring\n\t            var gh = ((((ga * ga) >>> 17) + ga * gb) >>> 15) + gb * gb;\n\t            var gl = (((gx & 0xffff0000) * gx) | 0) + (((gx & 0x0000ffff) * gx) | 0);\n\n\t            // High XOR low\n\t            G[i] = gh ^ gl;\n\t        }\n\n\t        // Calculate new state values\n\t        X[0] = (G[0] + ((G[7] << 16) | (G[7] >>> 16)) + ((G[6] << 16) | (G[6] >>> 16))) | 0;\n\t        X[1] = (G[1] + ((G[0] << 8)  | (G[0] >>> 24)) + G[7]) | 0;\n\t        X[2] = (G[2] + ((G[1] << 16) | (G[1] >>> 16)) + ((G[0] << 16) | (G[0] >>> 16))) | 0;\n\t        X[3] = (G[3] + ((G[2] << 8)  | (G[2] >>> 24)) + G[1]) | 0;\n\t        X[4] = (G[4] + ((G[3] << 16) | (G[3] >>> 16)) + ((G[2] << 16) | (G[2] >>> 16))) | 0;\n\t        X[5] = (G[5] + ((G[4] << 8)  | (G[4] >>> 24)) + G[3]) | 0;\n\t        X[6] = (G[6] + ((G[5] << 16) | (G[5] >>> 16)) + ((G[4] << 16) | (G[4] >>> 16))) | 0;\n\t        X[7] = (G[7] + ((G[6] << 8)  | (G[6] >>> 24)) + G[5]) | 0;\n\t    }\n\n\t    /**\n\t     * Shortcut functions to the cipher's object interface.\n\t     *\n\t     * @example\n\t     *\n\t     *     var ciphertext = CryptoJS.RabbitLegacy.encrypt(message, key, cfg);\n\t     *     var plaintext  = CryptoJS.RabbitLegacy.decrypt(ciphertext, key, cfg);\n\t     */\n\t    C.RabbitLegacy = StreamCipher._createHelper(RabbitLegacy);\n\t}());\n\n\n\treturn CryptoJS.RabbitLegacy;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./enc-base64\"), require(\"./md5\"), require(\"./evpkdf\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./enc-base64\", \"./md5\", \"./evpkdf\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var StreamCipher = C_lib.StreamCipher;\n\t    var C_algo = C.algo;\n\n\t    // Reusable objects\n\t    var S  = [];\n\t    var C_ = [];\n\t    var G  = [];\n\n\t    /**\n\t     * Rabbit stream cipher algorithm\n\t     */\n\t    var Rabbit = C_algo.Rabbit = StreamCipher.extend({\n\t        _doReset: function () {\n\t            // Shortcuts\n\t            var K = this._key.words;\n\t            var iv = this.cfg.iv;\n\n\t            // Swap endian\n\t            for (var i = 0; i < 4; i++) {\n\t                K[i] = (((K[i] << 8)  | (K[i] >>> 24)) & 0x00ff00ff) |\n\t                       (((K[i] << 24) | (K[i] >>> 8))  & 0xff00ff00);\n\t            }\n\n\t            // Generate initial state values\n\t            var X = this._X = [\n\t                K[0], (K[3] << 16) | (K[2] >>> 16),\n\t                K[1], (K[0] << 16) | (K[3] >>> 16),\n\t                K[2], (K[1] << 16) | (K[0] >>> 16),\n\t                K[3], (K[2] << 16) | (K[1] >>> 16)\n\t            ];\n\n\t            // Generate initial counter values\n\t            var C = this._C = [\n\t                (K[2] << 16) | (K[2] >>> 16), (K[0] & 0xffff0000) | (K[1] & 0x0000ffff),\n\t                (K[3] << 16) | (K[3] >>> 16), (K[1] & 0xffff0000) | (K[2] & 0x0000ffff),\n\t                (K[0] << 16) | (K[0] >>> 16), (K[2] & 0xffff0000) | (K[3] & 0x0000ffff),\n\t                (K[1] << 16) | (K[1] >>> 16), (K[3] & 0xffff0000) | (K[0] & 0x0000ffff)\n\t            ];\n\n\t            // Carry bit\n\t            this._b = 0;\n\n\t            // Iterate the system four times\n\t            for (var i = 0; i < 4; i++) {\n\t                nextState.call(this);\n\t            }\n\n\t            // Modify the counters\n\t            for (var i = 0; i < 8; i++) {\n\t                C[i] ^= X[(i + 4) & 7];\n\t            }\n\n\t            // IV setup\n\t            if (iv) {\n\t                // Shortcuts\n\t                var IV = iv.words;\n\t                var IV_0 = IV[0];\n\t                var IV_1 = IV[1];\n\n\t                // Generate four subvectors\n\t                var i0 = (((IV_0 << 8) | (IV_0 >>> 24)) & 0x00ff00ff) | (((IV_0 << 24) | (IV_0 >>> 8)) & 0xff00ff00);\n\t                var i2 = (((IV_1 << 8) | (IV_1 >>> 24)) & 0x00ff00ff) | (((IV_1 << 24) | (IV_1 >>> 8)) & 0xff00ff00);\n\t                var i1 = (i0 >>> 16) | (i2 & 0xffff0000);\n\t                var i3 = (i2 << 16)  | (i0 & 0x0000ffff);\n\n\t                // Modify counter values\n\t                C[0] ^= i0;\n\t                C[1] ^= i1;\n\t                C[2] ^= i2;\n\t                C[3] ^= i3;\n\t                C[4] ^= i0;\n\t                C[5] ^= i1;\n\t                C[6] ^= i2;\n\t                C[7] ^= i3;\n\n\t                // Iterate the system four times\n\t                for (var i = 0; i < 4; i++) {\n\t                    nextState.call(this);\n\t                }\n\t            }\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\t            // Shortcut\n\t            var X = this._X;\n\n\t            // Iterate the system\n\t            nextState.call(this);\n\n\t            // Generate four keystream words\n\t            S[0] = X[0] ^ (X[5] >>> 16) ^ (X[3] << 16);\n\t            S[1] = X[2] ^ (X[7] >>> 16) ^ (X[5] << 16);\n\t            S[2] = X[4] ^ (X[1] >>> 16) ^ (X[7] << 16);\n\t            S[3] = X[6] ^ (X[3] >>> 16) ^ (X[1] << 16);\n\n\t            for (var i = 0; i < 4; i++) {\n\t                // Swap endian\n\t                S[i] = (((S[i] << 8)  | (S[i] >>> 24)) & 0x00ff00ff) |\n\t                       (((S[i] << 24) | (S[i] >>> 8))  & 0xff00ff00);\n\n\t                // Encrypt\n\t                M[offset + i] ^= S[i];\n\t            }\n\t        },\n\n\t        blockSize: 128/32,\n\n\t        ivSize: 64/32\n\t    });\n\n\t    function nextState() {\n\t        // Shortcuts\n\t        var X = this._X;\n\t        var C = this._C;\n\n\t        // Save old counter values\n\t        for (var i = 0; i < 8; i++) {\n\t            C_[i] = C[i];\n\t        }\n\n\t        // Calculate new counter values\n\t        C[0] = (C[0] + 0x4d34d34d + this._b) | 0;\n\t        C[1] = (C[1] + 0xd34d34d3 + ((C[0] >>> 0) < (C_[0] >>> 0) ? 1 : 0)) | 0;\n\t        C[2] = (C[2] + 0x34d34d34 + ((C[1] >>> 0) < (C_[1] >>> 0) ? 1 : 0)) | 0;\n\t        C[3] = (C[3] + 0x4d34d34d + ((C[2] >>> 0) < (C_[2] >>> 0) ? 1 : 0)) | 0;\n\t        C[4] = (C[4] + 0xd34d34d3 + ((C[3] >>> 0) < (C_[3] >>> 0) ? 1 : 0)) | 0;\n\t        C[5] = (C[5] + 0x34d34d34 + ((C[4] >>> 0) < (C_[4] >>> 0) ? 1 : 0)) | 0;\n\t        C[6] = (C[6] + 0x4d34d34d + ((C[5] >>> 0) < (C_[5] >>> 0) ? 1 : 0)) | 0;\n\t        C[7] = (C[7] + 0xd34d34d3 + ((C[6] >>> 0) < (C_[6] >>> 0) ? 1 : 0)) | 0;\n\t        this._b = (C[7] >>> 0) < (C_[7] >>> 0) ? 1 : 0;\n\n\t        // Calculate the g-values\n\t        for (var i = 0; i < 8; i++) {\n\t            var gx = X[i] + C[i];\n\n\t            // Construct high and low argument for squaring\n\t            var ga = gx & 0xffff;\n\t            var gb = gx >>> 16;\n\n\t            // Calculate high and low result of squaring\n\t            var gh = ((((ga * ga) >>> 17) + ga * gb) >>> 15) + gb * gb;\n\t            var gl = (((gx & 0xffff0000) * gx) | 0) + (((gx & 0x0000ffff) * gx) | 0);\n\n\t            // High XOR low\n\t            G[i] = gh ^ gl;\n\t        }\n\n\t        // Calculate new state values\n\t        X[0] = (G[0] + ((G[7] << 16) | (G[7] >>> 16)) + ((G[6] << 16) | (G[6] >>> 16))) | 0;\n\t        X[1] = (G[1] + ((G[0] << 8)  | (G[0] >>> 24)) + G[7]) | 0;\n\t        X[2] = (G[2] + ((G[1] << 16) | (G[1] >>> 16)) + ((G[0] << 16) | (G[0] >>> 16))) | 0;\n\t        X[3] = (G[3] + ((G[2] << 8)  | (G[2] >>> 24)) + G[1]) | 0;\n\t        X[4] = (G[4] + ((G[3] << 16) | (G[3] >>> 16)) + ((G[2] << 16) | (G[2] >>> 16))) | 0;\n\t        X[5] = (G[5] + ((G[4] << 8)  | (G[4] >>> 24)) + G[3]) | 0;\n\t        X[6] = (G[6] + ((G[5] << 16) | (G[5] >>> 16)) + ((G[4] << 16) | (G[4] >>> 16))) | 0;\n\t        X[7] = (G[7] + ((G[6] << 8)  | (G[6] >>> 24)) + G[5]) | 0;\n\t    }\n\n\t    /**\n\t     * Shortcut functions to the cipher's object interface.\n\t     *\n\t     * @example\n\t     *\n\t     *     var ciphertext = CryptoJS.Rabbit.encrypt(message, key, cfg);\n\t     *     var plaintext  = CryptoJS.Rabbit.decrypt(ciphertext, key, cfg);\n\t     */\n\t    C.Rabbit = StreamCipher._createHelper(Rabbit);\n\t}());\n\n\n\treturn CryptoJS.Rabbit;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./enc-base64\"), require(\"./md5\"), require(\"./evpkdf\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./enc-base64\", \"./md5\", \"./evpkdf\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var StreamCipher = C_lib.StreamCipher;\n\t    var C_algo = C.algo;\n\n\t    /**\n\t     * RC4 stream cipher algorithm.\n\t     */\n\t    var RC4 = C_algo.RC4 = StreamCipher.extend({\n\t        _doReset: function () {\n\t            // Shortcuts\n\t            var key = this._key;\n\t            var keyWords = key.words;\n\t            var keySigBytes = key.sigBytes;\n\n\t            // Init sbox\n\t            var S = this._S = [];\n\t            for (var i = 0; i < 256; i++) {\n\t                S[i] = i;\n\t            }\n\n\t            // Key setup\n\t            for (var i = 0, j = 0; i < 256; i++) {\n\t                var keyByteIndex = i % keySigBytes;\n\t                var keyByte = (keyWords[keyByteIndex >>> 2] >>> (24 - (keyByteIndex % 4) * 8)) & 0xff;\n\n\t                j = (j + S[i] + keyByte) % 256;\n\n\t                // Swap\n\t                var t = S[i];\n\t                S[i] = S[j];\n\t                S[j] = t;\n\t            }\n\n\t            // Counters\n\t            this._i = this._j = 0;\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\t            M[offset] ^= generateKeystreamWord.call(this);\n\t        },\n\n\t        keySize: 256/32,\n\n\t        ivSize: 0\n\t    });\n\n\t    function generateKeystreamWord() {\n\t        // Shortcuts\n\t        var S = this._S;\n\t        var i = this._i;\n\t        var j = this._j;\n\n\t        // Generate keystream word\n\t        var keystreamWord = 0;\n\t        for (var n = 0; n < 4; n++) {\n\t            i = (i + 1) % 256;\n\t            j = (j + S[i]) % 256;\n\n\t            // Swap\n\t            var t = S[i];\n\t            S[i] = S[j];\n\t            S[j] = t;\n\n\t            keystreamWord |= S[(S[i] + S[j]) % 256] << (24 - n * 8);\n\t        }\n\n\t        // Update counters\n\t        this._i = i;\n\t        this._j = j;\n\n\t        return keystreamWord;\n\t    }\n\n\t    /**\n\t     * Shortcut functions to the cipher's object interface.\n\t     *\n\t     * @example\n\t     *\n\t     *     var ciphertext = CryptoJS.RC4.encrypt(message, key, cfg);\n\t     *     var plaintext  = CryptoJS.RC4.decrypt(ciphertext, key, cfg);\n\t     */\n\t    C.RC4 = StreamCipher._createHelper(RC4);\n\n\t    /**\n\t     * Modified RC4 stream cipher algorithm.\n\t     */\n\t    var RC4Drop = C_algo.RC4Drop = RC4.extend({\n\t        /**\n\t         * Configuration options.\n\t         *\n\t         * @property {number} drop The number of keystream words to drop. Default 192\n\t         */\n\t        cfg: RC4.cfg.extend({\n\t            drop: 192\n\t        }),\n\n\t        _doReset: function () {\n\t            RC4._doReset.call(this);\n\n\t            // Drop\n\t            for (var i = this.cfg.drop; i > 0; i--) {\n\t                generateKeystreamWord.call(this);\n\t            }\n\t        }\n\t    });\n\n\t    /**\n\t     * Shortcut functions to the cipher's object interface.\n\t     *\n\t     * @example\n\t     *\n\t     *     var ciphertext = CryptoJS.RC4Drop.encrypt(message, key, cfg);\n\t     *     var plaintext  = CryptoJS.RC4Drop.decrypt(ciphertext, key, cfg);\n\t     */\n\t    C.RC4Drop = StreamCipher._createHelper(RC4Drop);\n\t}());\n\n\n\treturn CryptoJS.RC4;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/** @preserve\n\t(c) 2012 by Cédric Mesnil. All rights reserved.\n\n\tRedistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:\n\n\t    - Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.\n\t    - Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.\n\n\tTHIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS \"AS IS\" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.\n\t*/\n\n\t(function (Math) {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var Hasher = C_lib.Hasher;\n\t    var C_algo = C.algo;\n\n\t    // Constants table\n\t    var _zl = WordArray.create([\n\t        0,  1,  2,  3,  4,  5,  6,  7,  8,  9, 10, 11, 12, 13, 14, 15,\n\t        7,  4, 13,  1, 10,  6, 15,  3, 12,  0,  9,  5,  2, 14, 11,  8,\n\t        3, 10, 14,  4,  9, 15,  8,  1,  2,  7,  0,  6, 13, 11,  5, 12,\n\t        1,  9, 11, 10,  0,  8, 12,  4, 13,  3,  7, 15, 14,  5,  6,  2,\n\t        4,  0,  5,  9,  7, 12,  2, 10, 14,  1,  3,  8, 11,  6, 15, 13]);\n\t    var _zr = WordArray.create([\n\t        5, 14,  7,  0,  9,  2, 11,  4, 13,  6, 15,  8,  1, 10,  3, 12,\n\t        6, 11,  3,  7,  0, 13,  5, 10, 14, 15,  8, 12,  4,  9,  1,  2,\n\t        15,  5,  1,  3,  7, 14,  6,  9, 11,  8, 12,  2, 10,  0,  4, 13,\n\t        8,  6,  4,  1,  3, 11, 15,  0,  5, 12,  2, 13,  9,  7, 10, 14,\n\t        12, 15, 10,  4,  1,  5,  8,  7,  6,  2, 13, 14,  0,  3,  9, 11]);\n\t    var _sl = WordArray.create([\n\t         11, 14, 15, 12,  5,  8,  7,  9, 11, 13, 14, 15,  6,  7,  9,  8,\n\t        7, 6,   8, 13, 11,  9,  7, 15,  7, 12, 15,  9, 11,  7, 13, 12,\n\t        11, 13,  6,  7, 14,  9, 13, 15, 14,  8, 13,  6,  5, 12,  7,  5,\n\t          11, 12, 14, 15, 14, 15,  9,  8,  9, 14,  5,  6,  8,  6,  5, 12,\n\t        9, 15,  5, 11,  6,  8, 13, 12,  5, 12, 13, 14, 11,  8,  5,  6 ]);\n\t    var _sr = WordArray.create([\n\t        8,  9,  9, 11, 13, 15, 15,  5,  7,  7,  8, 11, 14, 14, 12,  6,\n\t        9, 13, 15,  7, 12,  8,  9, 11,  7,  7, 12,  7,  6, 15, 13, 11,\n\t        9,  7, 15, 11,  8,  6,  6, 14, 12, 13,  5, 14, 13, 13,  7,  5,\n\t        15,  5,  8, 11, 14, 14,  6, 14,  6,  9, 12,  9, 12,  5, 15,  8,\n\t        8,  5, 12,  9, 12,  5, 14,  6,  8, 13,  6,  5, 15, 13, 11, 11 ]);\n\n\t    var _hl =  WordArray.create([ 0x00000000, 0x5A827999, 0x6ED9EBA1, 0x8F1BBCDC, 0xA953FD4E]);\n\t    var _hr =  WordArray.create([ 0x50A28BE6, 0x5C4DD124, 0x6D703EF3, 0x7A6D76E9, 0x00000000]);\n\n\t    /**\n\t     * RIPEMD160 hash algorithm.\n\t     */\n\t    var RIPEMD160 = C_algo.RIPEMD160 = Hasher.extend({\n\t        _doReset: function () {\n\t            this._hash  = WordArray.create([0x67452301, 0xEFCDAB89, 0x98BADCFE, 0x10325476, 0xC3D2E1F0]);\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\n\t            // Swap endian\n\t            for (var i = 0; i < 16; i++) {\n\t                // Shortcuts\n\t                var offset_i = offset + i;\n\t                var M_offset_i = M[offset_i];\n\n\t                // Swap\n\t                M[offset_i] = (\n\t                    (((M_offset_i << 8)  | (M_offset_i >>> 24)) & 0x00ff00ff) |\n\t                    (((M_offset_i << 24) | (M_offset_i >>> 8))  & 0xff00ff00)\n\t                );\n\t            }\n\t            // Shortcut\n\t            var H  = this._hash.words;\n\t            var hl = _hl.words;\n\t            var hr = _hr.words;\n\t            var zl = _zl.words;\n\t            var zr = _zr.words;\n\t            var sl = _sl.words;\n\t            var sr = _sr.words;\n\n\t            // Working variables\n\t            var al, bl, cl, dl, el;\n\t            var ar, br, cr, dr, er;\n\n\t            ar = al = H[0];\n\t            br = bl = H[1];\n\t            cr = cl = H[2];\n\t            dr = dl = H[3];\n\t            er = el = H[4];\n\t            // Computation\n\t            var t;\n\t            for (var i = 0; i < 80; i += 1) {\n\t                t = (al +  M[offset+zl[i]])|0;\n\t                if (i<16){\n\t\t            t +=  f1(bl,cl,dl) + hl[0];\n\t                } else if (i<32) {\n\t\t            t +=  f2(bl,cl,dl) + hl[1];\n\t                } else if (i<48) {\n\t\t            t +=  f3(bl,cl,dl) + hl[2];\n\t                } else if (i<64) {\n\t\t            t +=  f4(bl,cl,dl) + hl[3];\n\t                } else {// if (i<80) {\n\t\t            t +=  f5(bl,cl,dl) + hl[4];\n\t                }\n\t                t = t|0;\n\t                t =  rotl(t,sl[i]);\n\t                t = (t+el)|0;\n\t                al = el;\n\t                el = dl;\n\t                dl = rotl(cl, 10);\n\t                cl = bl;\n\t                bl = t;\n\n\t                t = (ar + M[offset+zr[i]])|0;\n\t                if (i<16){\n\t\t            t +=  f5(br,cr,dr) + hr[0];\n\t                } else if (i<32) {\n\t\t            t +=  f4(br,cr,dr) + hr[1];\n\t                } else if (i<48) {\n\t\t            t +=  f3(br,cr,dr) + hr[2];\n\t                } else if (i<64) {\n\t\t            t +=  f2(br,cr,dr) + hr[3];\n\t                } else {// if (i<80) {\n\t\t            t +=  f1(br,cr,dr) + hr[4];\n\t                }\n\t                t = t|0;\n\t                t =  rotl(t,sr[i]) ;\n\t                t = (t+er)|0;\n\t                ar = er;\n\t                er = dr;\n\t                dr = rotl(cr, 10);\n\t                cr = br;\n\t                br = t;\n\t            }\n\t            // Intermediate hash value\n\t            t    = (H[1] + cl + dr)|0;\n\t            H[1] = (H[2] + dl + er)|0;\n\t            H[2] = (H[3] + el + ar)|0;\n\t            H[3] = (H[4] + al + br)|0;\n\t            H[4] = (H[0] + bl + cr)|0;\n\t            H[0] =  t;\n\t        },\n\n\t        _doFinalize: function () {\n\t            // Shortcuts\n\t            var data = this._data;\n\t            var dataWords = data.words;\n\n\t            var nBitsTotal = this._nDataBytes * 8;\n\t            var nBitsLeft = data.sigBytes * 8;\n\n\t            // Add padding\n\t            dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);\n\t            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = (\n\t                (((nBitsTotal << 8)  | (nBitsTotal >>> 24)) & 0x00ff00ff) |\n\t                (((nBitsTotal << 24) | (nBitsTotal >>> 8))  & 0xff00ff00)\n\t            );\n\t            data.sigBytes = (dataWords.length + 1) * 4;\n\n\t            // Hash final blocks\n\t            this._process();\n\n\t            // Shortcuts\n\t            var hash = this._hash;\n\t            var H = hash.words;\n\n\t            // Swap endian\n\t            for (var i = 0; i < 5; i++) {\n\t                // Shortcut\n\t                var H_i = H[i];\n\n\t                // Swap\n\t                H[i] = (((H_i << 8)  | (H_i >>> 24)) & 0x00ff00ff) |\n\t                       (((H_i << 24) | (H_i >>> 8))  & 0xff00ff00);\n\t            }\n\n\t            // Return final computed hash\n\t            return hash;\n\t        },\n\n\t        clone: function () {\n\t            var clone = Hasher.clone.call(this);\n\t            clone._hash = this._hash.clone();\n\n\t            return clone;\n\t        }\n\t    });\n\n\n\t    function f1(x, y, z) {\n\t        return ((x) ^ (y) ^ (z));\n\n\t    }\n\n\t    function f2(x, y, z) {\n\t        return (((x)&(y)) | ((~x)&(z)));\n\t    }\n\n\t    function f3(x, y, z) {\n\t        return (((x) | (~(y))) ^ (z));\n\t    }\n\n\t    function f4(x, y, z) {\n\t        return (((x) & (z)) | ((y)&(~(z))));\n\t    }\n\n\t    function f5(x, y, z) {\n\t        return ((x) ^ ((y) |(~(z))));\n\n\t    }\n\n\t    function rotl(x,n) {\n\t        return (x<<n) | (x>>>(32-n));\n\t    }\n\n\n\t    /**\n\t     * Shortcut function to the hasher's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     *\n\t     * @return {WordArray} The hash.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hash = CryptoJS.RIPEMD160('message');\n\t     *     var hash = CryptoJS.RIPEMD160(wordArray);\n\t     */\n\t    C.RIPEMD160 = Hasher._createHelper(RIPEMD160);\n\n\t    /**\n\t     * Shortcut function to the HMAC's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     * @param {WordArray|string} key The secret key.\n\t     *\n\t     * @return {WordArray} The HMAC.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hmac = CryptoJS.HmacRIPEMD160(message, key);\n\t     */\n\t    C.HmacRIPEMD160 = Hasher._createHmacHelper(RIPEMD160);\n\t}(Math));\n\n\n\treturn CryptoJS.RIPEMD160;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var Hasher = C_lib.Hasher;\n\t    var C_algo = C.algo;\n\n\t    // Reusable object\n\t    var W = [];\n\n\t    /**\n\t     * SHA-1 hash algorithm.\n\t     */\n\t    var SHA1 = C_algo.SHA1 = Hasher.extend({\n\t        _doReset: function () {\n\t            this._hash = new WordArray.init([\n\t                0x67452301, 0xefcdab89,\n\t                0x98badcfe, 0x10325476,\n\t                0xc3d2e1f0\n\t            ]);\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\t            // Shortcut\n\t            var H = this._hash.words;\n\n\t            // Working variables\n\t            var a = H[0];\n\t            var b = H[1];\n\t            var c = H[2];\n\t            var d = H[3];\n\t            var e = H[4];\n\n\t            // Computation\n\t            for (var i = 0; i < 80; i++) {\n\t                if (i < 16) {\n\t                    W[i] = M[offset + i] | 0;\n\t                } else {\n\t                    var n = W[i - 3] ^ W[i - 8] ^ W[i - 14] ^ W[i - 16];\n\t                    W[i] = (n << 1) | (n >>> 31);\n\t                }\n\n\t                var t = ((a << 5) | (a >>> 27)) + e + W[i];\n\t                if (i < 20) {\n\t                    t += ((b & c) | (~b & d)) + 0x5a827999;\n\t                } else if (i < 40) {\n\t                    t += (b ^ c ^ d) + 0x6ed9eba1;\n\t                } else if (i < 60) {\n\t                    t += ((b & c) | (b & d) | (c & d)) - 0x70e44324;\n\t                } else /* if (i < 80) */ {\n\t                    t += (b ^ c ^ d) - 0x359d3e2a;\n\t                }\n\n\t                e = d;\n\t                d = c;\n\t                c = (b << 30) | (b >>> 2);\n\t                b = a;\n\t                a = t;\n\t            }\n\n\t            // Intermediate hash value\n\t            H[0] = (H[0] + a) | 0;\n\t            H[1] = (H[1] + b) | 0;\n\t            H[2] = (H[2] + c) | 0;\n\t            H[3] = (H[3] + d) | 0;\n\t            H[4] = (H[4] + e) | 0;\n\t        },\n\n\t        _doFinalize: function () {\n\t            // Shortcuts\n\t            var data = this._data;\n\t            var dataWords = data.words;\n\n\t            var nBitsTotal = this._nDataBytes * 8;\n\t            var nBitsLeft = data.sigBytes * 8;\n\n\t            // Add padding\n\t            dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);\n\t            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = Math.floor(nBitsTotal / 0x100000000);\n\t            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 15] = nBitsTotal;\n\t            data.sigBytes = dataWords.length * 4;\n\n\t            // Hash final blocks\n\t            this._process();\n\n\t            // Return final computed hash\n\t            return this._hash;\n\t        },\n\n\t        clone: function () {\n\t            var clone = Hasher.clone.call(this);\n\t            clone._hash = this._hash.clone();\n\n\t            return clone;\n\t        }\n\t    });\n\n\t    /**\n\t     * Shortcut function to the hasher's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     *\n\t     * @return {WordArray} The hash.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hash = CryptoJS.SHA1('message');\n\t     *     var hash = CryptoJS.SHA1(wordArray);\n\t     */\n\t    C.SHA1 = Hasher._createHelper(SHA1);\n\n\t    /**\n\t     * Shortcut function to the HMAC's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     * @param {WordArray|string} key The secret key.\n\t     *\n\t     * @return {WordArray} The HMAC.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hmac = CryptoJS.HmacSHA1(message, key);\n\t     */\n\t    C.HmacSHA1 = Hasher._createHmacHelper(SHA1);\n\t}());\n\n\n\treturn CryptoJS.SHA1;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./sha256\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./sha256\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var C_algo = C.algo;\n\t    var SHA256 = C_algo.SHA256;\n\n\t    /**\n\t     * SHA-224 hash algorithm.\n\t     */\n\t    var SHA224 = C_algo.SHA224 = SHA256.extend({\n\t        _doReset: function () {\n\t            this._hash = new WordArray.init([\n\t                0xc1059ed8, 0x367cd507, 0x3070dd17, 0xf70e5939,\n\t                0xffc00b31, 0x68581511, 0x64f98fa7, 0xbefa4fa4\n\t            ]);\n\t        },\n\n\t        _doFinalize: function () {\n\t            var hash = SHA256._doFinalize.call(this);\n\n\t            hash.sigBytes -= 4;\n\n\t            return hash;\n\t        }\n\t    });\n\n\t    /**\n\t     * Shortcut function to the hasher's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     *\n\t     * @return {WordArray} The hash.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hash = CryptoJS.SHA224('message');\n\t     *     var hash = CryptoJS.SHA224(wordArray);\n\t     */\n\t    C.SHA224 = SHA256._createHelper(SHA224);\n\n\t    /**\n\t     * Shortcut function to the HMAC's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     * @param {WordArray|string} key The secret key.\n\t     *\n\t     * @return {WordArray} The HMAC.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hmac = CryptoJS.HmacSHA224(message, key);\n\t     */\n\t    C.HmacSHA224 = SHA256._createHmacHelper(SHA224);\n\t}());\n\n\n\treturn CryptoJS.SHA224;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function (Math) {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var Hasher = C_lib.Hasher;\n\t    var C_algo = C.algo;\n\n\t    // Initialization and round constants tables\n\t    var H = [];\n\t    var K = [];\n\n\t    // Compute constants\n\t    (function () {\n\t        function isPrime(n) {\n\t            var sqrtN = Math.sqrt(n);\n\t            for (var factor = 2; factor <= sqrtN; factor++) {\n\t                if (!(n % factor)) {\n\t                    return false;\n\t                }\n\t            }\n\n\t            return true;\n\t        }\n\n\t        function getFractionalBits(n) {\n\t            return ((n - (n | 0)) * 0x100000000) | 0;\n\t        }\n\n\t        var n = 2;\n\t        var nPrime = 0;\n\t        while (nPrime < 64) {\n\t            if (isPrime(n)) {\n\t                if (nPrime < 8) {\n\t                    H[nPrime] = getFractionalBits(Math.pow(n, 1 / 2));\n\t                }\n\t                K[nPrime] = getFractionalBits(Math.pow(n, 1 / 3));\n\n\t                nPrime++;\n\t            }\n\n\t            n++;\n\t        }\n\t    }());\n\n\t    // Reusable object\n\t    var W = [];\n\n\t    /**\n\t     * SHA-256 hash algorithm.\n\t     */\n\t    var SHA256 = C_algo.SHA256 = Hasher.extend({\n\t        _doReset: function () {\n\t            this._hash = new WordArray.init(H.slice(0));\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\t            // Shortcut\n\t            var H = this._hash.words;\n\n\t            // Working variables\n\t            var a = H[0];\n\t            var b = H[1];\n\t            var c = H[2];\n\t            var d = H[3];\n\t            var e = H[4];\n\t            var f = H[5];\n\t            var g = H[6];\n\t            var h = H[7];\n\n\t            // Computation\n\t            for (var i = 0; i < 64; i++) {\n\t                if (i < 16) {\n\t                    W[i] = M[offset + i] | 0;\n\t                } else {\n\t                    var gamma0x = W[i - 15];\n\t                    var gamma0  = ((gamma0x << 25) | (gamma0x >>> 7))  ^\n\t                                  ((gamma0x << 14) | (gamma0x >>> 18)) ^\n\t                                   (gamma0x >>> 3);\n\n\t                    var gamma1x = W[i - 2];\n\t                    var gamma1  = ((gamma1x << 15) | (gamma1x >>> 17)) ^\n\t                                  ((gamma1x << 13) | (gamma1x >>> 19)) ^\n\t                                   (gamma1x >>> 10);\n\n\t                    W[i] = gamma0 + W[i - 7] + gamma1 + W[i - 16];\n\t                }\n\n\t                var ch  = (e & f) ^ (~e & g);\n\t                var maj = (a & b) ^ (a & c) ^ (b & c);\n\n\t                var sigma0 = ((a << 30) | (a >>> 2)) ^ ((a << 19) | (a >>> 13)) ^ ((a << 10) | (a >>> 22));\n\t                var sigma1 = ((e << 26) | (e >>> 6)) ^ ((e << 21) | (e >>> 11)) ^ ((e << 7)  | (e >>> 25));\n\n\t                var t1 = h + sigma1 + ch + K[i] + W[i];\n\t                var t2 = sigma0 + maj;\n\n\t                h = g;\n\t                g = f;\n\t                f = e;\n\t                e = (d + t1) | 0;\n\t                d = c;\n\t                c = b;\n\t                b = a;\n\t                a = (t1 + t2) | 0;\n\t            }\n\n\t            // Intermediate hash value\n\t            H[0] = (H[0] + a) | 0;\n\t            H[1] = (H[1] + b) | 0;\n\t            H[2] = (H[2] + c) | 0;\n\t            H[3] = (H[3] + d) | 0;\n\t            H[4] = (H[4] + e) | 0;\n\t            H[5] = (H[5] + f) | 0;\n\t            H[6] = (H[6] + g) | 0;\n\t            H[7] = (H[7] + h) | 0;\n\t        },\n\n\t        _doFinalize: function () {\n\t            // Shortcuts\n\t            var data = this._data;\n\t            var dataWords = data.words;\n\n\t            var nBitsTotal = this._nDataBytes * 8;\n\t            var nBitsLeft = data.sigBytes * 8;\n\n\t            // Add padding\n\t            dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);\n\t            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = Math.floor(nBitsTotal / 0x100000000);\n\t            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 15] = nBitsTotal;\n\t            data.sigBytes = dataWords.length * 4;\n\n\t            // Hash final blocks\n\t            this._process();\n\n\t            // Return final computed hash\n\t            return this._hash;\n\t        },\n\n\t        clone: function () {\n\t            var clone = Hasher.clone.call(this);\n\t            clone._hash = this._hash.clone();\n\n\t            return clone;\n\t        }\n\t    });\n\n\t    /**\n\t     * Shortcut function to the hasher's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     *\n\t     * @return {WordArray} The hash.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hash = CryptoJS.SHA256('message');\n\t     *     var hash = CryptoJS.SHA256(wordArray);\n\t     */\n\t    C.SHA256 = Hasher._createHelper(SHA256);\n\n\t    /**\n\t     * Shortcut function to the HMAC's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     * @param {WordArray|string} key The secret key.\n\t     *\n\t     * @return {WordArray} The HMAC.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hmac = CryptoJS.HmacSHA256(message, key);\n\t     */\n\t    C.HmacSHA256 = Hasher._createHmacHelper(SHA256);\n\t}(Math));\n\n\n\treturn CryptoJS.SHA256;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./x64-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./x64-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function (Math) {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var Hasher = C_lib.Hasher;\n\t    var C_x64 = C.x64;\n\t    var X64Word = C_x64.Word;\n\t    var C_algo = C.algo;\n\n\t    // Constants tables\n\t    var RHO_OFFSETS = [];\n\t    var PI_INDEXES  = [];\n\t    var ROUND_CONSTANTS = [];\n\n\t    // Compute Constants\n\t    (function () {\n\t        // Compute rho offset constants\n\t        var x = 1, y = 0;\n\t        for (var t = 0; t < 24; t++) {\n\t            RHO_OFFSETS[x + 5 * y] = ((t + 1) * (t + 2) / 2) % 64;\n\n\t            var newX = y % 5;\n\t            var newY = (2 * x + 3 * y) % 5;\n\t            x = newX;\n\t            y = newY;\n\t        }\n\n\t        // Compute pi index constants\n\t        for (var x = 0; x < 5; x++) {\n\t            for (var y = 0; y < 5; y++) {\n\t                PI_INDEXES[x + 5 * y] = y + ((2 * x + 3 * y) % 5) * 5;\n\t            }\n\t        }\n\n\t        // Compute round constants\n\t        var LFSR = 0x01;\n\t        for (var i = 0; i < 24; i++) {\n\t            var roundConstantMsw = 0;\n\t            var roundConstantLsw = 0;\n\n\t            for (var j = 0; j < 7; j++) {\n\t                if (LFSR & 0x01) {\n\t                    var bitPosition = (1 << j) - 1;\n\t                    if (bitPosition < 32) {\n\t                        roundConstantLsw ^= 1 << bitPosition;\n\t                    } else /* if (bitPosition >= 32) */ {\n\t                        roundConstantMsw ^= 1 << (bitPosition - 32);\n\t                    }\n\t                }\n\n\t                // Compute next LFSR\n\t                if (LFSR & 0x80) {\n\t                    // Primitive polynomial over GF(2): x^8 + x^6 + x^5 + x^4 + 1\n\t                    LFSR = (LFSR << 1) ^ 0x71;\n\t                } else {\n\t                    LFSR <<= 1;\n\t                }\n\t            }\n\n\t            ROUND_CONSTANTS[i] = X64Word.create(roundConstantMsw, roundConstantLsw);\n\t        }\n\t    }());\n\n\t    // Reusable objects for temporary values\n\t    var T = [];\n\t    (function () {\n\t        for (var i = 0; i < 25; i++) {\n\t            T[i] = X64Word.create();\n\t        }\n\t    }());\n\n\t    /**\n\t     * SHA-3 hash algorithm.\n\t     */\n\t    var SHA3 = C_algo.SHA3 = Hasher.extend({\n\t        /**\n\t         * Configuration options.\n\t         *\n\t         * @property {number} outputLength\n\t         *   The desired number of bits in the output hash.\n\t         *   Only values permitted are: 224, 256, 384, 512.\n\t         *   Default: 512\n\t         */\n\t        cfg: Hasher.cfg.extend({\n\t            outputLength: 512\n\t        }),\n\n\t        _doReset: function () {\n\t            var state = this._state = []\n\t            for (var i = 0; i < 25; i++) {\n\t                state[i] = new X64Word.init();\n\t            }\n\n\t            this.blockSize = (1600 - 2 * this.cfg.outputLength) / 32;\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\t            // Shortcuts\n\t            var state = this._state;\n\t            var nBlockSizeLanes = this.blockSize / 2;\n\n\t            // Absorb\n\t            for (var i = 0; i < nBlockSizeLanes; i++) {\n\t                // Shortcuts\n\t                var M2i  = M[offset + 2 * i];\n\t                var M2i1 = M[offset + 2 * i + 1];\n\n\t                // Swap endian\n\t                M2i = (\n\t                    (((M2i << 8)  | (M2i >>> 24)) & 0x00ff00ff) |\n\t                    (((M2i << 24) | (M2i >>> 8))  & 0xff00ff00)\n\t                );\n\t                M2i1 = (\n\t                    (((M2i1 << 8)  | (M2i1 >>> 24)) & 0x00ff00ff) |\n\t                    (((M2i1 << 24) | (M2i1 >>> 8))  & 0xff00ff00)\n\t                );\n\n\t                // Absorb message into state\n\t                var lane = state[i];\n\t                lane.high ^= M2i1;\n\t                lane.low  ^= M2i;\n\t            }\n\n\t            // Rounds\n\t            for (var round = 0; round < 24; round++) {\n\t                // Theta\n\t                for (var x = 0; x < 5; x++) {\n\t                    // Mix column lanes\n\t                    var tMsw = 0, tLsw = 0;\n\t                    for (var y = 0; y < 5; y++) {\n\t                        var lane = state[x + 5 * y];\n\t                        tMsw ^= lane.high;\n\t                        tLsw ^= lane.low;\n\t                    }\n\n\t                    // Temporary values\n\t                    var Tx = T[x];\n\t                    Tx.high = tMsw;\n\t                    Tx.low  = tLsw;\n\t                }\n\t                for (var x = 0; x < 5; x++) {\n\t                    // Shortcuts\n\t                    var Tx4 = T[(x + 4) % 5];\n\t                    var Tx1 = T[(x + 1) % 5];\n\t                    var Tx1Msw = Tx1.high;\n\t                    var Tx1Lsw = Tx1.low;\n\n\t                    // Mix surrounding columns\n\t                    var tMsw = Tx4.high ^ ((Tx1Msw << 1) | (Tx1Lsw >>> 31));\n\t                    var tLsw = Tx4.low  ^ ((Tx1Lsw << 1) | (Tx1Msw >>> 31));\n\t                    for (var y = 0; y < 5; y++) {\n\t                        var lane = state[x + 5 * y];\n\t                        lane.high ^= tMsw;\n\t                        lane.low  ^= tLsw;\n\t                    }\n\t                }\n\n\t                // Rho Pi\n\t                for (var laneIndex = 1; laneIndex < 25; laneIndex++) {\n\t                    // Shortcuts\n\t                    var lane = state[laneIndex];\n\t                    var laneMsw = lane.high;\n\t                    var laneLsw = lane.low;\n\t                    var rhoOffset = RHO_OFFSETS[laneIndex];\n\n\t                    // Rotate lanes\n\t                    if (rhoOffset < 32) {\n\t                        var tMsw = (laneMsw << rhoOffset) | (laneLsw >>> (32 - rhoOffset));\n\t                        var tLsw = (laneLsw << rhoOffset) | (laneMsw >>> (32 - rhoOffset));\n\t                    } else /* if (rhoOffset >= 32) */ {\n\t                        var tMsw = (laneLsw << (rhoOffset - 32)) | (laneMsw >>> (64 - rhoOffset));\n\t                        var tLsw = (laneMsw << (rhoOffset - 32)) | (laneLsw >>> (64 - rhoOffset));\n\t                    }\n\n\t                    // Transpose lanes\n\t                    var TPiLane = T[PI_INDEXES[laneIndex]];\n\t                    TPiLane.high = tMsw;\n\t                    TPiLane.low  = tLsw;\n\t                }\n\n\t                // Rho pi at x = y = 0\n\t                var T0 = T[0];\n\t                var state0 = state[0];\n\t                T0.high = state0.high;\n\t                T0.low  = state0.low;\n\n\t                // Chi\n\t                for (var x = 0; x < 5; x++) {\n\t                    for (var y = 0; y < 5; y++) {\n\t                        // Shortcuts\n\t                        var laneIndex = x + 5 * y;\n\t                        var lane = state[laneIndex];\n\t                        var TLane = T[laneIndex];\n\t                        var Tx1Lane = T[((x + 1) % 5) + 5 * y];\n\t                        var Tx2Lane = T[((x + 2) % 5) + 5 * y];\n\n\t                        // Mix rows\n\t                        lane.high = TLane.high ^ (~Tx1Lane.high & Tx2Lane.high);\n\t                        lane.low  = TLane.low  ^ (~Tx1Lane.low  & Tx2Lane.low);\n\t                    }\n\t                }\n\n\t                // Iota\n\t                var lane = state[0];\n\t                var roundConstant = ROUND_CONSTANTS[round];\n\t                lane.high ^= roundConstant.high;\n\t                lane.low  ^= roundConstant.low;;\n\t            }\n\t        },\n\n\t        _doFinalize: function () {\n\t            // Shortcuts\n\t            var data = this._data;\n\t            var dataWords = data.words;\n\t            var nBitsTotal = this._nDataBytes * 8;\n\t            var nBitsLeft = data.sigBytes * 8;\n\t            var blockSizeBits = this.blockSize * 32;\n\n\t            // Add padding\n\t            dataWords[nBitsLeft >>> 5] |= 0x1 << (24 - nBitsLeft % 32);\n\t            dataWords[((Math.ceil((nBitsLeft + 1) / blockSizeBits) * blockSizeBits) >>> 5) - 1] |= 0x80;\n\t            data.sigBytes = dataWords.length * 4;\n\n\t            // Hash final blocks\n\t            this._process();\n\n\t            // Shortcuts\n\t            var state = this._state;\n\t            var outputLengthBytes = this.cfg.outputLength / 8;\n\t            var outputLengthLanes = outputLengthBytes / 8;\n\n\t            // Squeeze\n\t            var hashWords = [];\n\t            for (var i = 0; i < outputLengthLanes; i++) {\n\t                // Shortcuts\n\t                var lane = state[i];\n\t                var laneMsw = lane.high;\n\t                var laneLsw = lane.low;\n\n\t                // Swap endian\n\t                laneMsw = (\n\t                    (((laneMsw << 8)  | (laneMsw >>> 24)) & 0x00ff00ff) |\n\t                    (((laneMsw << 24) | (laneMsw >>> 8))  & 0xff00ff00)\n\t                );\n\t                laneLsw = (\n\t                    (((laneLsw << 8)  | (laneLsw >>> 24)) & 0x00ff00ff) |\n\t                    (((laneLsw << 24) | (laneLsw >>> 8))  & 0xff00ff00)\n\t                );\n\n\t                // Squeeze state to retrieve hash\n\t                hashWords.push(laneLsw);\n\t                hashWords.push(laneMsw);\n\t            }\n\n\t            // Return final computed hash\n\t            return new WordArray.init(hashWords, outputLengthBytes);\n\t        },\n\n\t        clone: function () {\n\t            var clone = Hasher.clone.call(this);\n\n\t            var state = clone._state = this._state.slice(0);\n\t            for (var i = 0; i < 25; i++) {\n\t                state[i] = state[i].clone();\n\t            }\n\n\t            return clone;\n\t        }\n\t    });\n\n\t    /**\n\t     * Shortcut function to the hasher's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     *\n\t     * @return {WordArray} The hash.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hash = CryptoJS.SHA3('message');\n\t     *     var hash = CryptoJS.SHA3(wordArray);\n\t     */\n\t    C.SHA3 = Hasher._createHelper(SHA3);\n\n\t    /**\n\t     * Shortcut function to the HMAC's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     * @param {WordArray|string} key The secret key.\n\t     *\n\t     * @return {WordArray} The HMAC.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hmac = CryptoJS.HmacSHA3(message, key);\n\t     */\n\t    C.HmacSHA3 = Hasher._createHmacHelper(SHA3);\n\t}(Math));\n\n\n\treturn CryptoJS.SHA3;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./x64-core\"), require(\"./sha512\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./x64-core\", \"./sha512\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_x64 = C.x64;\n\t    var X64Word = C_x64.Word;\n\t    var X64WordArray = C_x64.WordArray;\n\t    var C_algo = C.algo;\n\t    var SHA512 = C_algo.SHA512;\n\n\t    /**\n\t     * SHA-384 hash algorithm.\n\t     */\n\t    var SHA384 = C_algo.SHA384 = SHA512.extend({\n\t        _doReset: function () {\n\t            this._hash = new X64WordArray.init([\n\t                new X64Word.init(0xcbbb9d5d, 0xc1059ed8), new X64Word.init(0x629a292a, 0x367cd507),\n\t                new X64Word.init(0x9159015a, 0x3070dd17), new X64Word.init(0x152fecd8, 0xf70e5939),\n\t                new X64Word.init(0x67332667, 0xffc00b31), new X64Word.init(0x8eb44a87, 0x68581511),\n\t                new X64Word.init(0xdb0c2e0d, 0x64f98fa7), new X64Word.init(0x47b5481d, 0xbefa4fa4)\n\t            ]);\n\t        },\n\n\t        _doFinalize: function () {\n\t            var hash = SHA512._doFinalize.call(this);\n\n\t            hash.sigBytes -= 16;\n\n\t            return hash;\n\t        }\n\t    });\n\n\t    /**\n\t     * Shortcut function to the hasher's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     *\n\t     * @return {WordArray} The hash.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hash = CryptoJS.SHA384('message');\n\t     *     var hash = CryptoJS.SHA384(wordArray);\n\t     */\n\t    C.SHA384 = SHA512._createHelper(SHA384);\n\n\t    /**\n\t     * Shortcut function to the HMAC's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     * @param {WordArray|string} key The secret key.\n\t     *\n\t     * @return {WordArray} The HMAC.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hmac = CryptoJS.HmacSHA384(message, key);\n\t     */\n\t    C.HmacSHA384 = SHA512._createHmacHelper(SHA384);\n\t}());\n\n\n\treturn CryptoJS.SHA384;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./x64-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./x64-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var Hasher = C_lib.Hasher;\n\t    var C_x64 = C.x64;\n\t    var X64Word = C_x64.Word;\n\t    var X64WordArray = C_x64.WordArray;\n\t    var C_algo = C.algo;\n\n\t    function X64Word_create() {\n\t        return X64Word.create.apply(X64Word, arguments);\n\t    }\n\n\t    // Constants\n\t    var K = [\n\t        X64Word_create(0x428a2f98, 0xd728ae22), X64Word_create(0x71374491, 0x23ef65cd),\n\t        X64Word_create(0xb5c0fbcf, 0xec4d3b2f), X64Word_create(0xe9b5dba5, 0x8189dbbc),\n\t        X64Word_create(0x3956c25b, 0xf348b538), X64Word_create(0x59f111f1, 0xb605d019),\n\t        X64Word_create(0x923f82a4, 0xaf194f9b), X64Word_create(0xab1c5ed5, 0xda6d8118),\n\t        X64Word_create(0xd807aa98, 0xa3030242), X64Word_create(0x12835b01, 0x45706fbe),\n\t        X64Word_create(0x243185be, 0x4ee4b28c), X64Word_create(0x550c7dc3, 0xd5ffb4e2),\n\t        X64Word_create(0x72be5d74, 0xf27b896f), X64Word_create(0x80deb1fe, 0x3b1696b1),\n\t        X64Word_create(0x9bdc06a7, 0x25c71235), X64Word_create(0xc19bf174, 0xcf692694),\n\t        X64Word_create(0xe49b69c1, 0x9ef14ad2), X64Word_create(0xefbe4786, 0x384f25e3),\n\t        X64Word_create(0x0fc19dc6, 0x8b8cd5b5), X64Word_create(0x240ca1cc, 0x77ac9c65),\n\t        X64Word_create(0x2de92c6f, 0x592b0275), X64Word_create(0x4a7484aa, 0x6ea6e483),\n\t        X64Word_create(0x5cb0a9dc, 0xbd41fbd4), X64Word_create(0x76f988da, 0x831153b5),\n\t        X64Word_create(0x983e5152, 0xee66dfab), X64Word_create(0xa831c66d, 0x2db43210),\n\t        X64Word_create(0xb00327c8, 0x98fb213f), X64Word_create(0xbf597fc7, 0xbeef0ee4),\n\t        X64Word_create(0xc6e00bf3, 0x3da88fc2), X64Word_create(0xd5a79147, 0x930aa725),\n\t        X64Word_create(0x06ca6351, 0xe003826f), X64Word_create(0x14292967, 0x0a0e6e70),\n\t        X64Word_create(0x27b70a85, 0x46d22ffc), X64Word_create(0x2e1b2138, 0x5c26c926),\n\t        X64Word_create(0x4d2c6dfc, 0x5ac42aed), X64Word_create(0x53380d13, 0x9d95b3df),\n\t        X64Word_create(0x650a7354, 0x8baf63de), X64Word_create(0x766a0abb, 0x3c77b2a8),\n\t        X64Word_create(0x81c2c92e, 0x47edaee6), X64Word_create(0x92722c85, 0x1482353b),\n\t        X64Word_create(0xa2bfe8a1, 0x4cf10364), X64Word_create(0xa81a664b, 0xbc423001),\n\t        X64Word_create(0xc24b8b70, 0xd0f89791), X64Word_create(0xc76c51a3, 0x0654be30),\n\t        X64Word_create(0xd192e819, 0xd6ef5218), X64Word_create(0xd6990624, 0x5565a910),\n\t        X64Word_create(0xf40e3585, 0x5771202a), X64Word_create(0x106aa070, 0x32bbd1b8),\n\t        X64Word_create(0x19a4c116, 0xb8d2d0c8), X64Word_create(0x1e376c08, 0x5141ab53),\n\t        X64Word_create(0x2748774c, 0xdf8eeb99), X64Word_create(0x34b0bcb5, 0xe19b48a8),\n\t        X64Word_create(0x391c0cb3, 0xc5c95a63), X64Word_create(0x4ed8aa4a, 0xe3418acb),\n\t        X64Word_create(0x5b9cca4f, 0x7763e373), X64Word_create(0x682e6ff3, 0xd6b2b8a3),\n\t        X64Word_create(0x748f82ee, 0x5defb2fc), X64Word_create(0x78a5636f, 0x43172f60),\n\t        X64Word_create(0x84c87814, 0xa1f0ab72), X64Word_create(0x8cc70208, 0x1a6439ec),\n\t        X64Word_create(0x90befffa, 0x23631e28), X64Word_create(0xa4506ceb, 0xde82bde9),\n\t        X64Word_create(0xbef9a3f7, 0xb2c67915), X64Word_create(0xc67178f2, 0xe372532b),\n\t        X64Word_create(0xca273ece, 0xea26619c), X64Word_create(0xd186b8c7, 0x21c0c207),\n\t        X64Word_create(0xeada7dd6, 0xcde0eb1e), X64Word_create(0xf57d4f7f, 0xee6ed178),\n\t        X64Word_create(0x06f067aa, 0x72176fba), X64Word_create(0x0a637dc5, 0xa2c898a6),\n\t        X64Word_create(0x113f9804, 0xbef90dae), X64Word_create(0x1b710b35, 0x131c471b),\n\t        X64Word_create(0x28db77f5, 0x23047d84), X64Word_create(0x32caab7b, 0x40c72493),\n\t        X64Word_create(0x3c9ebe0a, 0x15c9bebc), X64Word_create(0x431d67c4, 0x9c100d4c),\n\t        X64Word_create(0x4cc5d4be, 0xcb3e42b6), X64Word_create(0x597f299c, 0xfc657e2a),\n\t        X64Word_create(0x5fcb6fab, 0x3ad6faec), X64Word_create(0x6c44198c, 0x4a475817)\n\t    ];\n\n\t    // Reusable objects\n\t    var W = [];\n\t    (function () {\n\t        for (var i = 0; i < 80; i++) {\n\t            W[i] = X64Word_create();\n\t        }\n\t    }());\n\n\t    /**\n\t     * SHA-512 hash algorithm.\n\t     */\n\t    var SHA512 = C_algo.SHA512 = Hasher.extend({\n\t        _doReset: function () {\n\t            this._hash = new X64WordArray.init([\n\t                new X64Word.init(0x6a09e667, 0xf3bcc908), new X64Word.init(0xbb67ae85, 0x84caa73b),\n\t                new X64Word.init(0x3c6ef372, 0xfe94f82b), new X64Word.init(0xa54ff53a, 0x5f1d36f1),\n\t                new X64Word.init(0x510e527f, 0xade682d1), new X64Word.init(0x9b05688c, 0x2b3e6c1f),\n\t                new X64Word.init(0x1f83d9ab, 0xfb41bd6b), new X64Word.init(0x5be0cd19, 0x137e2179)\n\t            ]);\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\t            // Shortcuts\n\t            var H = this._hash.words;\n\n\t            var H0 = H[0];\n\t            var H1 = H[1];\n\t            var H2 = H[2];\n\t            var H3 = H[3];\n\t            var H4 = H[4];\n\t            var H5 = H[5];\n\t            var H6 = H[6];\n\t            var H7 = H[7];\n\n\t            var H0h = H0.high;\n\t            var H0l = H0.low;\n\t            var H1h = H1.high;\n\t            var H1l = H1.low;\n\t            var H2h = H2.high;\n\t            var H2l = H2.low;\n\t            var H3h = H3.high;\n\t            var H3l = H3.low;\n\t            var H4h = H4.high;\n\t            var H4l = H4.low;\n\t            var H5h = H5.high;\n\t            var H5l = H5.low;\n\t            var H6h = H6.high;\n\t            var H6l = H6.low;\n\t            var H7h = H7.high;\n\t            var H7l = H7.low;\n\n\t            // Working variables\n\t            var ah = H0h;\n\t            var al = H0l;\n\t            var bh = H1h;\n\t            var bl = H1l;\n\t            var ch = H2h;\n\t            var cl = H2l;\n\t            var dh = H3h;\n\t            var dl = H3l;\n\t            var eh = H4h;\n\t            var el = H4l;\n\t            var fh = H5h;\n\t            var fl = H5l;\n\t            var gh = H6h;\n\t            var gl = H6l;\n\t            var hh = H7h;\n\t            var hl = H7l;\n\n\t            // Rounds\n\t            for (var i = 0; i < 80; i++) {\n\t                // Shortcut\n\t                var Wi = W[i];\n\n\t                // Extend message\n\t                if (i < 16) {\n\t                    var Wih = Wi.high = M[offset + i * 2]     | 0;\n\t                    var Wil = Wi.low  = M[offset + i * 2 + 1] | 0;\n\t                } else {\n\t                    // Gamma0\n\t                    var gamma0x  = W[i - 15];\n\t                    var gamma0xh = gamma0x.high;\n\t                    var gamma0xl = gamma0x.low;\n\t                    var gamma0h  = ((gamma0xh >>> 1) | (gamma0xl << 31)) ^ ((gamma0xh >>> 8) | (gamma0xl << 24)) ^ (gamma0xh >>> 7);\n\t                    var gamma0l  = ((gamma0xl >>> 1) | (gamma0xh << 31)) ^ ((gamma0xl >>> 8) | (gamma0xh << 24)) ^ ((gamma0xl >>> 7) | (gamma0xh << 25));\n\n\t                    // Gamma1\n\t                    var gamma1x  = W[i - 2];\n\t                    var gamma1xh = gamma1x.high;\n\t                    var gamma1xl = gamma1x.low;\n\t                    var gamma1h  = ((gamma1xh >>> 19) | (gamma1xl << 13)) ^ ((gamma1xh << 3) | (gamma1xl >>> 29)) ^ (gamma1xh >>> 6);\n\t                    var gamma1l  = ((gamma1xl >>> 19) | (gamma1xh << 13)) ^ ((gamma1xl << 3) | (gamma1xh >>> 29)) ^ ((gamma1xl >>> 6) | (gamma1xh << 26));\n\n\t                    // W[i] = gamma0 + W[i - 7] + gamma1 + W[i - 16]\n\t                    var Wi7  = W[i - 7];\n\t                    var Wi7h = Wi7.high;\n\t                    var Wi7l = Wi7.low;\n\n\t                    var Wi16  = W[i - 16];\n\t                    var Wi16h = Wi16.high;\n\t                    var Wi16l = Wi16.low;\n\n\t                    var Wil = gamma0l + Wi7l;\n\t                    var Wih = gamma0h + Wi7h + ((Wil >>> 0) < (gamma0l >>> 0) ? 1 : 0);\n\t                    var Wil = Wil + gamma1l;\n\t                    var Wih = Wih + gamma1h + ((Wil >>> 0) < (gamma1l >>> 0) ? 1 : 0);\n\t                    var Wil = Wil + Wi16l;\n\t                    var Wih = Wih + Wi16h + ((Wil >>> 0) < (Wi16l >>> 0) ? 1 : 0);\n\n\t                    Wi.high = Wih;\n\t                    Wi.low  = Wil;\n\t                }\n\n\t                var chh  = (eh & fh) ^ (~eh & gh);\n\t                var chl  = (el & fl) ^ (~el & gl);\n\t                var majh = (ah & bh) ^ (ah & ch) ^ (bh & ch);\n\t                var majl = (al & bl) ^ (al & cl) ^ (bl & cl);\n\n\t                var sigma0h = ((ah >>> 28) | (al << 4))  ^ ((ah << 30)  | (al >>> 2)) ^ ((ah << 25) | (al >>> 7));\n\t                var sigma0l = ((al >>> 28) | (ah << 4))  ^ ((al << 30)  | (ah >>> 2)) ^ ((al << 25) | (ah >>> 7));\n\t                var sigma1h = ((eh >>> 14) | (el << 18)) ^ ((eh >>> 18) | (el << 14)) ^ ((eh << 23) | (el >>> 9));\n\t                var sigma1l = ((el >>> 14) | (eh << 18)) ^ ((el >>> 18) | (eh << 14)) ^ ((el << 23) | (eh >>> 9));\n\n\t                // t1 = h + sigma1 + ch + K[i] + W[i]\n\t                var Ki  = K[i];\n\t                var Kih = Ki.high;\n\t                var Kil = Ki.low;\n\n\t                var t1l = hl + sigma1l;\n\t                var t1h = hh + sigma1h + ((t1l >>> 0) < (hl >>> 0) ? 1 : 0);\n\t                var t1l = t1l + chl;\n\t                var t1h = t1h + chh + ((t1l >>> 0) < (chl >>> 0) ? 1 : 0);\n\t                var t1l = t1l + Kil;\n\t                var t1h = t1h + Kih + ((t1l >>> 0) < (Kil >>> 0) ? 1 : 0);\n\t                var t1l = t1l + Wil;\n\t                var t1h = t1h + Wih + ((t1l >>> 0) < (Wil >>> 0) ? 1 : 0);\n\n\t                // t2 = sigma0 + maj\n\t                var t2l = sigma0l + majl;\n\t                var t2h = sigma0h + majh + ((t2l >>> 0) < (sigma0l >>> 0) ? 1 : 0);\n\n\t                // Update working variables\n\t                hh = gh;\n\t                hl = gl;\n\t                gh = fh;\n\t                gl = fl;\n\t                fh = eh;\n\t                fl = el;\n\t                el = (dl + t1l) | 0;\n\t                eh = (dh + t1h + ((el >>> 0) < (dl >>> 0) ? 1 : 0)) | 0;\n\t                dh = ch;\n\t                dl = cl;\n\t                ch = bh;\n\t                cl = bl;\n\t                bh = ah;\n\t                bl = al;\n\t                al = (t1l + t2l) | 0;\n\t                ah = (t1h + t2h + ((al >>> 0) < (t1l >>> 0) ? 1 : 0)) | 0;\n\t            }\n\n\t            // Intermediate hash value\n\t            H0l = H0.low  = (H0l + al);\n\t            H0.high = (H0h + ah + ((H0l >>> 0) < (al >>> 0) ? 1 : 0));\n\t            H1l = H1.low  = (H1l + bl);\n\t            H1.high = (H1h + bh + ((H1l >>> 0) < (bl >>> 0) ? 1 : 0));\n\t            H2l = H2.low  = (H2l + cl);\n\t            H2.high = (H2h + ch + ((H2l >>> 0) < (cl >>> 0) ? 1 : 0));\n\t            H3l = H3.low  = (H3l + dl);\n\t            H3.high = (H3h + dh + ((H3l >>> 0) < (dl >>> 0) ? 1 : 0));\n\t            H4l = H4.low  = (H4l + el);\n\t            H4.high = (H4h + eh + ((H4l >>> 0) < (el >>> 0) ? 1 : 0));\n\t            H5l = H5.low  = (H5l + fl);\n\t            H5.high = (H5h + fh + ((H5l >>> 0) < (fl >>> 0) ? 1 : 0));\n\t            H6l = H6.low  = (H6l + gl);\n\t            H6.high = (H6h + gh + ((H6l >>> 0) < (gl >>> 0) ? 1 : 0));\n\t            H7l = H7.low  = (H7l + hl);\n\t            H7.high = (H7h + hh + ((H7l >>> 0) < (hl >>> 0) ? 1 : 0));\n\t        },\n\n\t        _doFinalize: function () {\n\t            // Shortcuts\n\t            var data = this._data;\n\t            var dataWords = data.words;\n\n\t            var nBitsTotal = this._nDataBytes * 8;\n\t            var nBitsLeft = data.sigBytes * 8;\n\n\t            // Add padding\n\t            dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);\n\t            dataWords[(((nBitsLeft + 128) >>> 10) << 5) + 30] = Math.floor(nBitsTotal / 0x100000000);\n\t            dataWords[(((nBitsLeft + 128) >>> 10) << 5) + 31] = nBitsTotal;\n\t            data.sigBytes = dataWords.length * 4;\n\n\t            // Hash final blocks\n\t            this._process();\n\n\t            // Convert hash to 32-bit word array before returning\n\t            var hash = this._hash.toX32();\n\n\t            // Return final computed hash\n\t            return hash;\n\t        },\n\n\t        clone: function () {\n\t            var clone = Hasher.clone.call(this);\n\t            clone._hash = this._hash.clone();\n\n\t            return clone;\n\t        },\n\n\t        blockSize: 1024/32\n\t    });\n\n\t    /**\n\t     * Shortcut function to the hasher's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     *\n\t     * @return {WordArray} The hash.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hash = CryptoJS.SHA512('message');\n\t     *     var hash = CryptoJS.SHA512(wordArray);\n\t     */\n\t    C.SHA512 = Hasher._createHelper(SHA512);\n\n\t    /**\n\t     * Shortcut function to the HMAC's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     * @param {WordArray|string} key The secret key.\n\t     *\n\t     * @return {WordArray} The HMAC.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hmac = CryptoJS.HmacSHA512(message, key);\n\t     */\n\t    C.HmacSHA512 = Hasher._createHmacHelper(SHA512);\n\t}());\n\n\n\treturn CryptoJS.SHA512;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./enc-base64\"), require(\"./md5\"), require(\"./evpkdf\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./enc-base64\", \"./md5\", \"./evpkdf\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var BlockCipher = C_lib.BlockCipher;\n\t    var C_algo = C.algo;\n\n\t    // Permuted Choice 1 constants\n\t    var PC1 = [\n\t        57, 49, 41, 33, 25, 17, 9,  1,\n\t        58, 50, 42, 34, 26, 18, 10, 2,\n\t        59, 51, 43, 35, 27, 19, 11, 3,\n\t        60, 52, 44, 36, 63, 55, 47, 39,\n\t        31, 23, 15, 7,  62, 54, 46, 38,\n\t        30, 22, 14, 6,  61, 53, 45, 37,\n\t        29, 21, 13, 5,  28, 20, 12, 4\n\t    ];\n\n\t    // Permuted Choice 2 constants\n\t    var PC2 = [\n\t        14, 17, 11, 24, 1,  5,\n\t        3,  28, 15, 6,  21, 10,\n\t        23, 19, 12, 4,  26, 8,\n\t        16, 7,  27, 20, 13, 2,\n\t        41, 52, 31, 37, 47, 55,\n\t        30, 40, 51, 45, 33, 48,\n\t        44, 49, 39, 56, 34, 53,\n\t        46, 42, 50, 36, 29, 32\n\t    ];\n\n\t    // Cumulative bit shift constants\n\t    var BIT_SHIFTS = [1,  2,  4,  6,  8,  10, 12, 14, 15, 17, 19, 21, 23, 25, 27, 28];\n\n\t    // SBOXes and round permutation constants\n\t    var SBOX_P = [\n\t        {\n\t            0x0: 0x808200,\n\t            0x10000000: 0x8000,\n\t            0x20000000: 0x808002,\n\t            0x30000000: 0x2,\n\t            0x40000000: 0x200,\n\t            0x50000000: 0x808202,\n\t            0x60000000: 0x800202,\n\t            0x70000000: 0x800000,\n\t            0x80000000: 0x202,\n\t            0x90000000: 0x800200,\n\t            0xa0000000: 0x8200,\n\t            0xb0000000: 0x808000,\n\t            0xc0000000: 0x8002,\n\t            0xd0000000: 0x800002,\n\t            0xe0000000: 0x0,\n\t            0xf0000000: 0x8202,\n\t            0x8000000: 0x0,\n\t            0x18000000: 0x808202,\n\t            0x28000000: 0x8202,\n\t            0x38000000: 0x8000,\n\t            0x48000000: 0x808200,\n\t            0x58000000: 0x200,\n\t            0x68000000: 0x808002,\n\t            0x78000000: 0x2,\n\t            0x88000000: 0x800200,\n\t            0x98000000: 0x8200,\n\t            0xa8000000: 0x808000,\n\t            0xb8000000: 0x800202,\n\t            0xc8000000: 0x800002,\n\t            0xd8000000: 0x8002,\n\t            0xe8000000: 0x202,\n\t            0xf8000000: 0x800000,\n\t            0x1: 0x8000,\n\t            0x10000001: 0x2,\n\t            0x20000001: 0x808200,\n\t            0x30000001: 0x800000,\n\t            0x40000001: 0x808002,\n\t            0x50000001: 0x8200,\n\t            0x60000001: 0x200,\n\t            0x70000001: 0x800202,\n\t            0x80000001: 0x808202,\n\t            0x90000001: 0x808000,\n\t            0xa0000001: 0x800002,\n\t            0xb0000001: 0x8202,\n\t            0xc0000001: 0x202,\n\t            0xd0000001: 0x800200,\n\t            0xe0000001: 0x8002,\n\t            0xf0000001: 0x0,\n\t            0x8000001: 0x808202,\n\t            0x18000001: 0x808000,\n\t            0x28000001: 0x800000,\n\t            0x38000001: 0x200,\n\t            0x48000001: 0x8000,\n\t            0x58000001: 0x800002,\n\t            0x68000001: 0x2,\n\t            0x78000001: 0x8202,\n\t            0x88000001: 0x8002,\n\t            0x98000001: 0x800202,\n\t            0xa8000001: 0x202,\n\t            0xb8000001: 0x808200,\n\t            0xc8000001: 0x800200,\n\t            0xd8000001: 0x0,\n\t            0xe8000001: 0x8200,\n\t            0xf8000001: 0x808002\n\t        },\n\t        {\n\t            0x0: 0x40084010,\n\t            0x1000000: 0x4000,\n\t            0x2000000: 0x80000,\n\t            0x3000000: 0x40080010,\n\t            0x4000000: 0x40000010,\n\t            0x5000000: 0x40084000,\n\t            0x6000000: 0x40004000,\n\t            0x7000000: 0x10,\n\t            0x8000000: 0x84000,\n\t            0x9000000: 0x40004010,\n\t            0xa000000: 0x40000000,\n\t            0xb000000: 0x84010,\n\t            0xc000000: 0x80010,\n\t            0xd000000: 0x0,\n\t            0xe000000: 0x4010,\n\t            0xf000000: 0x40080000,\n\t            0x800000: 0x40004000,\n\t            0x1800000: 0x84010,\n\t            0x2800000: 0x10,\n\t            0x3800000: 0x40004010,\n\t            0x4800000: 0x40084010,\n\t            0x5800000: 0x40000000,\n\t            0x6800000: 0x80000,\n\t            0x7800000: 0x40080010,\n\t            0x8800000: 0x80010,\n\t            0x9800000: 0x0,\n\t            0xa800000: 0x4000,\n\t            0xb800000: 0x40080000,\n\t            0xc800000: 0x40000010,\n\t            0xd800000: 0x84000,\n\t            0xe800000: 0x40084000,\n\t            0xf800000: 0x4010,\n\t            0x10000000: 0x0,\n\t            0x11000000: 0x40080010,\n\t            0x12000000: 0x40004010,\n\t            0x13000000: 0x40084000,\n\t            0x14000000: 0x40080000,\n\t            0x15000000: 0x10,\n\t            0x16000000: 0x84010,\n\t            0x17000000: 0x4000,\n\t            0x18000000: 0x4010,\n\t            0x19000000: 0x80000,\n\t            0x1a000000: 0x80010,\n\t            0x1b000000: 0x40000010,\n\t            0x1c000000: 0x84000,\n\t            0x1d000000: 0x40004000,\n\t            0x1e000000: 0x40000000,\n\t            0x1f000000: 0x40084010,\n\t            0x10800000: 0x84010,\n\t            0x11800000: 0x80000,\n\t            0x12800000: 0x40080000,\n\t            0x13800000: 0x4000,\n\t            0x14800000: 0x40004000,\n\t            0x15800000: 0x40084010,\n\t            0x16800000: 0x10,\n\t            0x17800000: 0x40000000,\n\t            0x18800000: 0x40084000,\n\t            0x19800000: 0x40000010,\n\t            0x1a800000: 0x40004010,\n\t            0x1b800000: 0x80010,\n\t            0x1c800000: 0x0,\n\t            0x1d800000: 0x4010,\n\t            0x1e800000: 0x40080010,\n\t            0x1f800000: 0x84000\n\t        },\n\t        {\n\t            0x0: 0x104,\n\t            0x100000: 0x0,\n\t            0x200000: 0x4000100,\n\t            0x300000: 0x10104,\n\t            0x400000: 0x10004,\n\t            0x500000: 0x4000004,\n\t            0x600000: 0x4010104,\n\t            0x700000: 0x4010000,\n\t            0x800000: 0x4000000,\n\t            0x900000: 0x4010100,\n\t            0xa00000: 0x10100,\n\t            0xb00000: 0x4010004,\n\t            0xc00000: 0x4000104,\n\t            0xd00000: 0x10000,\n\t            0xe00000: 0x4,\n\t            0xf00000: 0x100,\n\t            0x80000: 0x4010100,\n\t            0x180000: 0x4010004,\n\t            0x280000: 0x0,\n\t            0x380000: 0x4000100,\n\t            0x480000: 0x4000004,\n\t            0x580000: 0x10000,\n\t            0x680000: 0x10004,\n\t            0x780000: 0x104,\n\t            0x880000: 0x4,\n\t            0x980000: 0x100,\n\t            0xa80000: 0x4010000,\n\t            0xb80000: 0x10104,\n\t            0xc80000: 0x10100,\n\t            0xd80000: 0x4000104,\n\t            0xe80000: 0x4010104,\n\t            0xf80000: 0x4000000,\n\t            0x1000000: 0x4010100,\n\t            0x1100000: 0x10004,\n\t            0x1200000: 0x10000,\n\t            0x1300000: 0x4000100,\n\t            0x1400000: 0x100,\n\t            0x1500000: 0x4010104,\n\t            0x1600000: 0x4000004,\n\t            0x1700000: 0x0,\n\t            0x1800000: 0x4000104,\n\t            0x1900000: 0x4000000,\n\t            0x1a00000: 0x4,\n\t            0x1b00000: 0x10100,\n\t            0x1c00000: 0x4010000,\n\t            0x1d00000: 0x104,\n\t            0x1e00000: 0x10104,\n\t            0x1f00000: 0x4010004,\n\t            0x1080000: 0x4000000,\n\t            0x1180000: 0x104,\n\t            0x1280000: 0x4010100,\n\t            0x1380000: 0x0,\n\t            0x1480000: 0x10004,\n\t            0x1580000: 0x4000100,\n\t            0x1680000: 0x100,\n\t            0x1780000: 0x4010004,\n\t            0x1880000: 0x10000,\n\t            0x1980000: 0x4010104,\n\t            0x1a80000: 0x10104,\n\t            0x1b80000: 0x4000004,\n\t            0x1c80000: 0x4000104,\n\t            0x1d80000: 0x4010000,\n\t            0x1e80000: 0x4,\n\t            0x1f80000: 0x10100\n\t        },\n\t        {\n\t            0x0: 0x80401000,\n\t            0x10000: 0x80001040,\n\t            0x20000: 0x401040,\n\t            0x30000: 0x80400000,\n\t            0x40000: 0x0,\n\t            0x50000: 0x401000,\n\t            0x60000: 0x80000040,\n\t            0x70000: 0x400040,\n\t            0x80000: 0x80000000,\n\t            0x90000: 0x400000,\n\t            0xa0000: 0x40,\n\t            0xb0000: 0x80001000,\n\t            0xc0000: 0x80400040,\n\t            0xd0000: 0x1040,\n\t            0xe0000: 0x1000,\n\t            0xf0000: 0x80401040,\n\t            0x8000: 0x80001040,\n\t            0x18000: 0x40,\n\t            0x28000: 0x80400040,\n\t            0x38000: 0x80001000,\n\t            0x48000: 0x401000,\n\t            0x58000: 0x80401040,\n\t            0x68000: 0x0,\n\t            0x78000: 0x80400000,\n\t            0x88000: 0x1000,\n\t            0x98000: 0x80401000,\n\t            0xa8000: 0x400000,\n\t            0xb8000: 0x1040,\n\t            0xc8000: 0x80000000,\n\t            0xd8000: 0x400040,\n\t            0xe8000: 0x401040,\n\t            0xf8000: 0x80000040,\n\t            0x100000: 0x400040,\n\t            0x110000: 0x401000,\n\t            0x120000: 0x80000040,\n\t            0x130000: 0x0,\n\t            0x140000: 0x1040,\n\t            0x150000: 0x80400040,\n\t            0x160000: 0x80401000,\n\t            0x170000: 0x80001040,\n\t            0x180000: 0x80401040,\n\t            0x190000: 0x80000000,\n\t            0x1a0000: 0x80400000,\n\t            0x1b0000: 0x401040,\n\t            0x1c0000: 0x80001000,\n\t            0x1d0000: 0x400000,\n\t            0x1e0000: 0x40,\n\t            0x1f0000: 0x1000,\n\t            0x108000: 0x80400000,\n\t            0x118000: 0x80401040,\n\t            0x128000: 0x0,\n\t            0x138000: 0x401000,\n\t            0x148000: 0x400040,\n\t            0x158000: 0x80000000,\n\t            0x168000: 0x80001040,\n\t            0x178000: 0x40,\n\t            0x188000: 0x80000040,\n\t            0x198000: 0x1000,\n\t            0x1a8000: 0x80001000,\n\t            0x1b8000: 0x80400040,\n\t            0x1c8000: 0x1040,\n\t            0x1d8000: 0x80401000,\n\t            0x1e8000: 0x400000,\n\t            0x1f8000: 0x401040\n\t        },\n\t        {\n\t            0x0: 0x80,\n\t            0x1000: 0x1040000,\n\t            0x2000: 0x40000,\n\t            0x3000: 0x20000000,\n\t            0x4000: 0x20040080,\n\t            0x5000: 0x1000080,\n\t            0x6000: 0x21000080,\n\t            0x7000: 0x40080,\n\t            0x8000: 0x1000000,\n\t            0x9000: 0x20040000,\n\t            0xa000: 0x20000080,\n\t            0xb000: 0x21040080,\n\t            0xc000: 0x21040000,\n\t            0xd000: 0x0,\n\t            0xe000: 0x1040080,\n\t            0xf000: 0x21000000,\n\t            0x800: 0x1040080,\n\t            0x1800: 0x21000080,\n\t            0x2800: 0x80,\n\t            0x3800: 0x1040000,\n\t            0x4800: 0x40000,\n\t            0x5800: 0x20040080,\n\t            0x6800: 0x21040000,\n\t            0x7800: 0x20000000,\n\t            0x8800: 0x20040000,\n\t            0x9800: 0x0,\n\t            0xa800: 0x21040080,\n\t            0xb800: 0x1000080,\n\t            0xc800: 0x20000080,\n\t            0xd800: 0x21000000,\n\t            0xe800: 0x1000000,\n\t            0xf800: 0x40080,\n\t            0x10000: 0x40000,\n\t            0x11000: 0x80,\n\t            0x12000: 0x20000000,\n\t            0x13000: 0x21000080,\n\t            0x14000: 0x1000080,\n\t            0x15000: 0x21040000,\n\t            0x16000: 0x20040080,\n\t            0x17000: 0x1000000,\n\t            0x18000: 0x21040080,\n\t            0x19000: 0x21000000,\n\t            0x1a000: 0x1040000,\n\t            0x1b000: 0x20040000,\n\t            0x1c000: 0x40080,\n\t            0x1d000: 0x20000080,\n\t            0x1e000: 0x0,\n\t            0x1f000: 0x1040080,\n\t            0x10800: 0x21000080,\n\t            0x11800: 0x1000000,\n\t            0x12800: 0x1040000,\n\t            0x13800: 0x20040080,\n\t            0x14800: 0x20000000,\n\t            0x15800: 0x1040080,\n\t            0x16800: 0x80,\n\t            0x17800: 0x21040000,\n\t            0x18800: 0x40080,\n\t            0x19800: 0x21040080,\n\t            0x1a800: 0x0,\n\t            0x1b800: 0x21000000,\n\t            0x1c800: 0x1000080,\n\t            0x1d800: 0x40000,\n\t            0x1e800: 0x20040000,\n\t            0x1f800: 0x20000080\n\t        },\n\t        {\n\t            0x0: 0x10000008,\n\t            0x100: 0x2000,\n\t            0x200: 0x10200000,\n\t            0x300: 0x10202008,\n\t            0x400: 0x10002000,\n\t            0x500: 0x200000,\n\t            0x600: 0x200008,\n\t            0x700: 0x10000000,\n\t            0x800: 0x0,\n\t            0x900: 0x10002008,\n\t            0xa00: 0x202000,\n\t            0xb00: 0x8,\n\t            0xc00: 0x10200008,\n\t            0xd00: 0x202008,\n\t            0xe00: 0x2008,\n\t            0xf00: 0x10202000,\n\t            0x80: 0x10200000,\n\t            0x180: 0x10202008,\n\t            0x280: 0x8,\n\t            0x380: 0x200000,\n\t            0x480: 0x202008,\n\t            0x580: 0x10000008,\n\t            0x680: 0x10002000,\n\t            0x780: 0x2008,\n\t            0x880: 0x200008,\n\t            0x980: 0x2000,\n\t            0xa80: 0x10002008,\n\t            0xb80: 0x10200008,\n\t            0xc80: 0x0,\n\t            0xd80: 0x10202000,\n\t            0xe80: 0x202000,\n\t            0xf80: 0x10000000,\n\t            0x1000: 0x10002000,\n\t            0x1100: 0x10200008,\n\t            0x1200: 0x10202008,\n\t            0x1300: 0x2008,\n\t            0x1400: 0x200000,\n\t            0x1500: 0x10000000,\n\t            0x1600: 0x10000008,\n\t            0x1700: 0x202000,\n\t            0x1800: 0x202008,\n\t            0x1900: 0x0,\n\t            0x1a00: 0x8,\n\t            0x1b00: 0x10200000,\n\t            0x1c00: 0x2000,\n\t            0x1d00: 0x10002008,\n\t            0x1e00: 0x10202000,\n\t            0x1f00: 0x200008,\n\t            0x1080: 0x8,\n\t            0x1180: 0x202000,\n\t            0x1280: 0x200000,\n\t            0x1380: 0x10000008,\n\t            0x1480: 0x10002000,\n\t            0x1580: 0x2008,\n\t            0x1680: 0x10202008,\n\t            0x1780: 0x10200000,\n\t            0x1880: 0x10202000,\n\t            0x1980: 0x10200008,\n\t            0x1a80: 0x2000,\n\t            0x1b80: 0x202008,\n\t            0x1c80: 0x200008,\n\t            0x1d80: 0x0,\n\t            0x1e80: 0x10000000,\n\t            0x1f80: 0x10002008\n\t        },\n\t        {\n\t            0x0: 0x100000,\n\t            0x10: 0x2000401,\n\t            0x20: 0x400,\n\t            0x30: 0x100401,\n\t            0x40: 0x2100401,\n\t            0x50: 0x0,\n\t            0x60: 0x1,\n\t            0x70: 0x2100001,\n\t            0x80: 0x2000400,\n\t            0x90: 0x100001,\n\t            0xa0: 0x2000001,\n\t            0xb0: 0x2100400,\n\t            0xc0: 0x2100000,\n\t            0xd0: 0x401,\n\t            0xe0: 0x100400,\n\t            0xf0: 0x2000000,\n\t            0x8: 0x2100001,\n\t            0x18: 0x0,\n\t            0x28: 0x2000401,\n\t            0x38: 0x2100400,\n\t            0x48: 0x100000,\n\t            0x58: 0x2000001,\n\t            0x68: 0x2000000,\n\t            0x78: 0x401,\n\t            0x88: 0x100401,\n\t            0x98: 0x2000400,\n\t            0xa8: 0x2100000,\n\t            0xb8: 0x100001,\n\t            0xc8: 0x400,\n\t            0xd8: 0x2100401,\n\t            0xe8: 0x1,\n\t            0xf8: 0x100400,\n\t            0x100: 0x2000000,\n\t            0x110: 0x100000,\n\t            0x120: 0x2000401,\n\t            0x130: 0x2100001,\n\t            0x140: 0x100001,\n\t            0x150: 0x2000400,\n\t            0x160: 0x2100400,\n\t            0x170: 0x100401,\n\t            0x180: 0x401,\n\t            0x190: 0x2100401,\n\t            0x1a0: 0x100400,\n\t            0x1b0: 0x1,\n\t            0x1c0: 0x0,\n\t            0x1d0: 0x2100000,\n\t            0x1e0: 0x2000001,\n\t            0x1f0: 0x400,\n\t            0x108: 0x100400,\n\t            0x118: 0x2000401,\n\t            0x128: 0x2100001,\n\t            0x138: 0x1,\n\t            0x148: 0x2000000,\n\t            0x158: 0x100000,\n\t            0x168: 0x401,\n\t            0x178: 0x2100400,\n\t            0x188: 0x2000001,\n\t            0x198: 0x2100000,\n\t            0x1a8: 0x0,\n\t            0x1b8: 0x2100401,\n\t            0x1c8: 0x100401,\n\t            0x1d8: 0x400,\n\t            0x1e8: 0x2000400,\n\t            0x1f8: 0x100001\n\t        },\n\t        {\n\t            0x0: 0x8000820,\n\t            0x1: 0x20000,\n\t            0x2: 0x8000000,\n\t            0x3: 0x20,\n\t            0x4: 0x20020,\n\t            0x5: 0x8020820,\n\t            0x6: 0x8020800,\n\t            0x7: 0x800,\n\t            0x8: 0x8020000,\n\t            0x9: 0x8000800,\n\t            0xa: 0x20800,\n\t            0xb: 0x8020020,\n\t            0xc: 0x820,\n\t            0xd: 0x0,\n\t            0xe: 0x8000020,\n\t            0xf: 0x20820,\n\t            0x80000000: 0x800,\n\t            0x80000001: 0x8020820,\n\t            0x80000002: 0x8000820,\n\t            0x80000003: 0x8000000,\n\t            0x80000004: 0x8020000,\n\t            0x80000005: 0x20800,\n\t            0x80000006: 0x20820,\n\t            0x80000007: 0x20,\n\t            0x80000008: 0x8000020,\n\t            0x80000009: 0x820,\n\t            0x8000000a: 0x20020,\n\t            0x8000000b: 0x8020800,\n\t            0x8000000c: 0x0,\n\t            0x8000000d: 0x8020020,\n\t            0x8000000e: 0x8000800,\n\t            0x8000000f: 0x20000,\n\t            0x10: 0x20820,\n\t            0x11: 0x8020800,\n\t            0x12: 0x20,\n\t            0x13: 0x800,\n\t            0x14: 0x8000800,\n\t            0x15: 0x8000020,\n\t            0x16: 0x8020020,\n\t            0x17: 0x20000,\n\t            0x18: 0x0,\n\t            0x19: 0x20020,\n\t            0x1a: 0x8020000,\n\t            0x1b: 0x8000820,\n\t            0x1c: 0x8020820,\n\t            0x1d: 0x20800,\n\t            0x1e: 0x820,\n\t            0x1f: 0x8000000,\n\t            0x80000010: 0x20000,\n\t            0x80000011: 0x800,\n\t            0x80000012: 0x8020020,\n\t            0x80000013: 0x20820,\n\t            0x80000014: 0x20,\n\t            0x80000015: 0x8020000,\n\t            0x80000016: 0x8000000,\n\t            0x80000017: 0x8000820,\n\t            0x80000018: 0x8020820,\n\t            0x80000019: 0x8000020,\n\t            0x8000001a: 0x8000800,\n\t            0x8000001b: 0x0,\n\t            0x8000001c: 0x20800,\n\t            0x8000001d: 0x820,\n\t            0x8000001e: 0x20020,\n\t            0x8000001f: 0x8020800\n\t        }\n\t    ];\n\n\t    // Masks that select the SBOX input\n\t    var SBOX_MASK = [\n\t        0xf8000001, 0x1f800000, 0x01f80000, 0x001f8000,\n\t        0x0001f800, 0x00001f80, 0x000001f8, 0x8000001f\n\t    ];\n\n\t    /**\n\t     * DES block cipher algorithm.\n\t     */\n\t    var DES = C_algo.DES = BlockCipher.extend({\n\t        _doReset: function () {\n\t            // Shortcuts\n\t            var key = this._key;\n\t            var keyWords = key.words;\n\n\t            // Select 56 bits according to PC1\n\t            var keyBits = [];\n\t            for (var i = 0; i < 56; i++) {\n\t                var keyBitPos = PC1[i] - 1;\n\t                keyBits[i] = (keyWords[keyBitPos >>> 5] >>> (31 - keyBitPos % 32)) & 1;\n\t            }\n\n\t            // Assemble 16 subkeys\n\t            var subKeys = this._subKeys = [];\n\t            for (var nSubKey = 0; nSubKey < 16; nSubKey++) {\n\t                // Create subkey\n\t                var subKey = subKeys[nSubKey] = [];\n\n\t                // Shortcut\n\t                var bitShift = BIT_SHIFTS[nSubKey];\n\n\t                // Select 48 bits according to PC2\n\t                for (var i = 0; i < 24; i++) {\n\t                    // Select from the left 28 key bits\n\t                    subKey[(i / 6) | 0] |= keyBits[((PC2[i] - 1) + bitShift) % 28] << (31 - i % 6);\n\n\t                    // Select from the right 28 key bits\n\t                    subKey[4 + ((i / 6) | 0)] |= keyBits[28 + (((PC2[i + 24] - 1) + bitShift) % 28)] << (31 - i % 6);\n\t                }\n\n\t                // Since each subkey is applied to an expanded 32-bit input,\n\t                // the subkey can be broken into 8 values scaled to 32-bits,\n\t                // which allows the key to be used without expansion\n\t                subKey[0] = (subKey[0] << 1) | (subKey[0] >>> 31);\n\t                for (var i = 1; i < 7; i++) {\n\t                    subKey[i] = subKey[i] >>> ((i - 1) * 4 + 3);\n\t                }\n\t                subKey[7] = (subKey[7] << 5) | (subKey[7] >>> 27);\n\t            }\n\n\t            // Compute inverse subkeys\n\t            var invSubKeys = this._invSubKeys = [];\n\t            for (var i = 0; i < 16; i++) {\n\t                invSubKeys[i] = subKeys[15 - i];\n\t            }\n\t        },\n\n\t        encryptBlock: function (M, offset) {\n\t            this._doCryptBlock(M, offset, this._subKeys);\n\t        },\n\n\t        decryptBlock: function (M, offset) {\n\t            this._doCryptBlock(M, offset, this._invSubKeys);\n\t        },\n\n\t        _doCryptBlock: function (M, offset, subKeys) {\n\t            // Get input\n\t            this._lBlock = M[offset];\n\t            this._rBlock = M[offset + 1];\n\n\t            // Initial permutation\n\t            exchangeLR.call(this, 4,  0x0f0f0f0f);\n\t            exchangeLR.call(this, 16, 0x0000ffff);\n\t            exchangeRL.call(this, 2,  0x33333333);\n\t            exchangeRL.call(this, 8,  0x00ff00ff);\n\t            exchangeLR.call(this, 1,  0x55555555);\n\n\t            // Rounds\n\t            for (var round = 0; round < 16; round++) {\n\t                // Shortcuts\n\t                var subKey = subKeys[round];\n\t                var lBlock = this._lBlock;\n\t                var rBlock = this._rBlock;\n\n\t                // Feistel function\n\t                var f = 0;\n\t                for (var i = 0; i < 8; i++) {\n\t                    f |= SBOX_P[i][((rBlock ^ subKey[i]) & SBOX_MASK[i]) >>> 0];\n\t                }\n\t                this._lBlock = rBlock;\n\t                this._rBlock = lBlock ^ f;\n\t            }\n\n\t            // Undo swap from last round\n\t            var t = this._lBlock;\n\t            this._lBlock = this._rBlock;\n\t            this._rBlock = t;\n\n\t            // Final permutation\n\t            exchangeLR.call(this, 1,  0x55555555);\n\t            exchangeRL.call(this, 8,  0x00ff00ff);\n\t            exchangeRL.call(this, 2,  0x33333333);\n\t            exchangeLR.call(this, 16, 0x0000ffff);\n\t            exchangeLR.call(this, 4,  0x0f0f0f0f);\n\n\t            // Set output\n\t            M[offset] = this._lBlock;\n\t            M[offset + 1] = this._rBlock;\n\t        },\n\n\t        keySize: 64/32,\n\n\t        ivSize: 64/32,\n\n\t        blockSize: 64/32\n\t    });\n\n\t    // Swap bits across the left and right words\n\t    function exchangeLR(offset, mask) {\n\t        var t = ((this._lBlock >>> offset) ^ this._rBlock) & mask;\n\t        this._rBlock ^= t;\n\t        this._lBlock ^= t << offset;\n\t    }\n\n\t    function exchangeRL(offset, mask) {\n\t        var t = ((this._rBlock >>> offset) ^ this._lBlock) & mask;\n\t        this._lBlock ^= t;\n\t        this._rBlock ^= t << offset;\n\t    }\n\n\t    /**\n\t     * Shortcut functions to the cipher's object interface.\n\t     *\n\t     * @example\n\t     *\n\t     *     var ciphertext = CryptoJS.DES.encrypt(message, key, cfg);\n\t     *     var plaintext  = CryptoJS.DES.decrypt(ciphertext, key, cfg);\n\t     */\n\t    C.DES = BlockCipher._createHelper(DES);\n\n\t    /**\n\t     * Triple-DES block cipher algorithm.\n\t     */\n\t    var TripleDES = C_algo.TripleDES = BlockCipher.extend({\n\t        _doReset: function () {\n\t            // Shortcuts\n\t            var key = this._key;\n\t            var keyWords = key.words;\n\n\t            // Create DES instances\n\t            this._des1 = DES.createEncryptor(WordArray.create(keyWords.slice(0, 2)));\n\t            this._des2 = DES.createEncryptor(WordArray.create(keyWords.slice(2, 4)));\n\t            this._des3 = DES.createEncryptor(WordArray.create(keyWords.slice(4, 6)));\n\t        },\n\n\t        encryptBlock: function (M, offset) {\n\t            this._des1.encryptBlock(M, offset);\n\t            this._des2.decryptBlock(M, offset);\n\t            this._des3.encryptBlock(M, offset);\n\t        },\n\n\t        decryptBlock: function (M, offset) {\n\t            this._des3.decryptBlock(M, offset);\n\t            this._des2.encryptBlock(M, offset);\n\t            this._des1.decryptBlock(M, offset);\n\t        },\n\n\t        keySize: 192/32,\n\n\t        ivSize: 64/32,\n\n\t        blockSize: 64/32\n\t    });\n\n\t    /**\n\t     * Shortcut functions to the cipher's object interface.\n\t     *\n\t     * @example\n\t     *\n\t     *     var ciphertext = CryptoJS.TripleDES.encrypt(message, key, cfg);\n\t     *     var plaintext  = CryptoJS.TripleDES.decrypt(ciphertext, key, cfg);\n\t     */\n\t    C.TripleDES = BlockCipher._createHelper(TripleDES);\n\t}());\n\n\n\treturn CryptoJS.TripleDES;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function (undefined) {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var Base = C_lib.Base;\n\t    var X32WordArray = C_lib.WordArray;\n\n\t    /**\n\t     * x64 namespace.\n\t     */\n\t    var C_x64 = C.x64 = {};\n\n\t    /**\n\t     * A 64-bit word.\n\t     */\n\t    var X64Word = C_x64.Word = Base.extend({\n\t        /**\n\t         * Initializes a newly created 64-bit word.\n\t         *\n\t         * @param {number} high The high 32 bits.\n\t         * @param {number} low The low 32 bits.\n\t         *\n\t         * @example\n\t         *\n\t         *     var x64Word = CryptoJS.x64.Word.create(0x00010203, 0x04050607);\n\t         */\n\t        init: function (high, low) {\n\t            this.high = high;\n\t            this.low = low;\n\t        }\n\n\t        /**\n\t         * Bitwise NOTs this word.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after negating.\n\t         *\n\t         * @example\n\t         *\n\t         *     var negated = x64Word.not();\n\t         */\n\t        // not: function () {\n\t            // var high = ~this.high;\n\t            // var low = ~this.low;\n\n\t            // return X64Word.create(high, low);\n\t        // },\n\n\t        /**\n\t         * Bitwise ANDs this word with the passed word.\n\t         *\n\t         * @param {X64Word} word The x64-Word to AND with this word.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after ANDing.\n\t         *\n\t         * @example\n\t         *\n\t         *     var anded = x64Word.and(anotherX64Word);\n\t         */\n\t        // and: function (word) {\n\t            // var high = this.high & word.high;\n\t            // var low = this.low & word.low;\n\n\t            // return X64Word.create(high, low);\n\t        // },\n\n\t        /**\n\t         * Bitwise ORs this word with the passed word.\n\t         *\n\t         * @param {X64Word} word The x64-Word to OR with this word.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after ORing.\n\t         *\n\t         * @example\n\t         *\n\t         *     var ored = x64Word.or(anotherX64Word);\n\t         */\n\t        // or: function (word) {\n\t            // var high = this.high | word.high;\n\t            // var low = this.low | word.low;\n\n\t            // return X64Word.create(high, low);\n\t        // },\n\n\t        /**\n\t         * Bitwise XORs this word with the passed word.\n\t         *\n\t         * @param {X64Word} word The x64-Word to XOR with this word.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after XORing.\n\t         *\n\t         * @example\n\t         *\n\t         *     var xored = x64Word.xor(anotherX64Word);\n\t         */\n\t        // xor: function (word) {\n\t            // var high = this.high ^ word.high;\n\t            // var low = this.low ^ word.low;\n\n\t            // return X64Word.create(high, low);\n\t        // },\n\n\t        /**\n\t         * Shifts this word n bits to the left.\n\t         *\n\t         * @param {number} n The number of bits to shift.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after shifting.\n\t         *\n\t         * @example\n\t         *\n\t         *     var shifted = x64Word.shiftL(25);\n\t         */\n\t        // shiftL: function (n) {\n\t            // if (n < 32) {\n\t                // var high = (this.high << n) | (this.low >>> (32 - n));\n\t                // var low = this.low << n;\n\t            // } else {\n\t                // var high = this.low << (n - 32);\n\t                // var low = 0;\n\t            // }\n\n\t            // return X64Word.create(high, low);\n\t        // },\n\n\t        /**\n\t         * Shifts this word n bits to the right.\n\t         *\n\t         * @param {number} n The number of bits to shift.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after shifting.\n\t         *\n\t         * @example\n\t         *\n\t         *     var shifted = x64Word.shiftR(7);\n\t         */\n\t        // shiftR: function (n) {\n\t            // if (n < 32) {\n\t                // var low = (this.low >>> n) | (this.high << (32 - n));\n\t                // var high = this.high >>> n;\n\t            // } else {\n\t                // var low = this.high >>> (n - 32);\n\t                // var high = 0;\n\t            // }\n\n\t            // return X64Word.create(high, low);\n\t        // },\n\n\t        /**\n\t         * Rotates this word n bits to the left.\n\t         *\n\t         * @param {number} n The number of bits to rotate.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after rotating.\n\t         *\n\t         * @example\n\t         *\n\t         *     var rotated = x64Word.rotL(25);\n\t         */\n\t        // rotL: function (n) {\n\t            // return this.shiftL(n).or(this.shiftR(64 - n));\n\t        // },\n\n\t        /**\n\t         * Rotates this word n bits to the right.\n\t         *\n\t         * @param {number} n The number of bits to rotate.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after rotating.\n\t         *\n\t         * @example\n\t         *\n\t         *     var rotated = x64Word.rotR(7);\n\t         */\n\t        // rotR: function (n) {\n\t            // return this.shiftR(n).or(this.shiftL(64 - n));\n\t        // },\n\n\t        /**\n\t         * Adds this word with the passed word.\n\t         *\n\t         * @param {X64Word} word The x64-Word to add with this word.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after adding.\n\t         *\n\t         * @example\n\t         *\n\t         *     var added = x64Word.add(anotherX64Word);\n\t         */\n\t        // add: function (word) {\n\t            // var low = (this.low + word.low) | 0;\n\t            // var carry = (low >>> 0) < (this.low >>> 0) ? 1 : 0;\n\t            // var high = (this.high + word.high + carry) | 0;\n\n\t            // return X64Word.create(high, low);\n\t        // }\n\t    });\n\n\t    /**\n\t     * An array of 64-bit words.\n\t     *\n\t     * @property {Array} words The array of CryptoJS.x64.Word objects.\n\t     * @property {number} sigBytes The number of significant bytes in this word array.\n\t     */\n\t    var X64WordArray = C_x64.WordArray = Base.extend({\n\t        /**\n\t         * Initializes a newly created word array.\n\t         *\n\t         * @param {Array} words (Optional) An array of CryptoJS.x64.Word objects.\n\t         * @param {number} sigBytes (Optional) The number of significant bytes in the words.\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.x64.WordArray.create();\n\t         *\n\t         *     var wordArray = CryptoJS.x64.WordArray.create([\n\t         *         CryptoJS.x64.Word.create(0x00010203, 0x04050607),\n\t         *         CryptoJS.x64.Word.create(0x18191a1b, 0x1c1d1e1f)\n\t         *     ]);\n\t         *\n\t         *     var wordArray = CryptoJS.x64.WordArray.create([\n\t         *         CryptoJS.x64.Word.create(0x00010203, 0x04050607),\n\t         *         CryptoJS.x64.Word.create(0x18191a1b, 0x1c1d1e1f)\n\t         *     ], 10);\n\t         */\n\t        init: function (words, sigBytes) {\n\t            words = this.words = words || [];\n\n\t            if (sigBytes != undefined) {\n\t                this.sigBytes = sigBytes;\n\t            } else {\n\t                this.sigBytes = words.length * 8;\n\t            }\n\t        },\n\n\t        /**\n\t         * Converts this 64-bit word array to a 32-bit word array.\n\t         *\n\t         * @return {CryptoJS.lib.WordArray} This word array's data as a 32-bit word array.\n\t         *\n\t         * @example\n\t         *\n\t         *     var x32WordArray = x64WordArray.toX32();\n\t         */\n\t        toX32: function () {\n\t            // Shortcuts\n\t            var x64Words = this.words;\n\t            var x64WordsLength = x64Words.length;\n\n\t            // Convert\n\t            var x32Words = [];\n\t            for (var i = 0; i < x64WordsLength; i++) {\n\t                var x64Word = x64Words[i];\n\t                x32Words.push(x64Word.high);\n\t                x32Words.push(x64Word.low);\n\t            }\n\n\t            return X32WordArray.create(x32Words, this.sigBytes);\n\t        },\n\n\t        /**\n\t         * Creates a copy of this word array.\n\t         *\n\t         * @return {X64WordArray} The clone.\n\t         *\n\t         * @example\n\t         *\n\t         *     var clone = x64WordArray.clone();\n\t         */\n\t        clone: function () {\n\t            var clone = Base.clone.call(this);\n\n\t            // Clone \"words\" array\n\t            var words = clone.words = this.words.slice(0);\n\n\t            // Clone each X64Word object\n\t            var wordsLength = words.length;\n\t            for (var i = 0; i < wordsLength; i++) {\n\t                words[i] = words[i].clone();\n\t            }\n\n\t            return clone;\n\t        }\n\t    });\n\t}());\n\n\n\treturn CryptoJS;\n\n}));","(function (self) {\n  'use strict';\n\n  function fetchPonyfill(options) {\n    var Promise = options && options.Promise || self.Promise;\n    var XMLHttpRequest = options && options.XMLHttpRequest || self.XMLHttpRequest;\n    var global = self;\n\n    return (function () {\n      var self = Object.create(global, {\n        fetch: {\n          value: undefined,\n          writable: true\n        }\n      });\n\n      (function(self) {\n        'use strict';\n\n        if (self.fetch) {\n          return\n        }\n\n        var support = {\n          searchParams: 'URLSearchParams' in self,\n          iterable: 'Symbol' in self && 'iterator' in Symbol,\n          blob: 'FileReader' in self && 'Blob' in self && (function() {\n            try {\n              new Blob()\n              return true\n            } catch(e) {\n              return false\n            }\n          })(),\n          formData: 'FormData' in self,\n          arrayBuffer: 'ArrayBuffer' in self\n        }\n\n        if (support.arrayBuffer) {\n          var viewClasses = [\n            '[object Int8Array]',\n            '[object Uint8Array]',\n            '[object Uint8ClampedArray]',\n            '[object Int16Array]',\n            '[object Uint16Array]',\n            '[object Int32Array]',\n            '[object Uint32Array]',\n            '[object Float32Array]',\n            '[object Float64Array]'\n          ]\n\n          var isDataView = function(obj) {\n            return obj && DataView.prototype.isPrototypeOf(obj)\n          }\n\n          var isArrayBufferView = ArrayBuffer.isView || function(obj) {\n            return obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1\n          }\n        }\n\n        function normalizeName(name) {\n          if (typeof name !== 'string') {\n            name = String(name)\n          }\n          if (/[^a-z0-9\\-#$%&'*+.\\^_`|~]/i.test(name)) {\n            throw new TypeError('Invalid character in header field name')\n          }\n          return name.toLowerCase()\n        }\n\n        function normalizeValue(value) {\n          if (typeof value !== 'string') {\n            value = String(value)\n          }\n          return value\n        }\n\n        // Build a destructive iterator for the value list\n        function iteratorFor(items) {\n          var iterator = {\n            next: function() {\n              var value = items.shift()\n              return {done: value === undefined, value: value}\n            }\n          }\n\n          if (support.iterable) {\n            iterator[Symbol.iterator] = function() {\n              return iterator\n            }\n          }\n\n          return iterator\n        }\n\n        function Headers(headers) {\n          this.map = {}\n\n          if (headers instanceof Headers) {\n            headers.forEach(function(value, name) {\n              this.append(name, value)\n            }, this)\n          } else if (Array.isArray(headers)) {\n            headers.forEach(function(header) {\n              this.append(header[0], header[1])\n            }, this)\n          } else if (headers) {\n            Object.getOwnPropertyNames(headers).forEach(function(name) {\n              this.append(name, headers[name])\n            }, this)\n          }\n        }\n\n        Headers.prototype.append = function(name, value) {\n          name = normalizeName(name)\n          value = normalizeValue(value)\n          var oldValue = this.map[name]\n          this.map[name] = oldValue ? oldValue+','+value : value\n        }\n\n        Headers.prototype['delete'] = function(name) {\n          delete this.map[normalizeName(name)]\n        }\n\n        Headers.prototype.get = function(name) {\n          name = normalizeName(name)\n          return this.has(name) ? this.map[name] : null\n        }\n\n        Headers.prototype.has = function(name) {\n          return this.map.hasOwnProperty(normalizeName(name))\n        }\n\n        Headers.prototype.set = function(name, value) {\n          this.map[normalizeName(name)] = normalizeValue(value)\n        }\n\n        Headers.prototype.forEach = function(callback, thisArg) {\n          for (var name in this.map) {\n            if (this.map.hasOwnProperty(name)) {\n              callback.call(thisArg, this.map[name], name, this)\n            }\n          }\n        }\n\n        Headers.prototype.keys = function() {\n          var items = []\n          this.forEach(function(value, name) { items.push(name) })\n          return iteratorFor(items)\n        }\n\n        Headers.prototype.values = function() {\n          var items = []\n          this.forEach(function(value) { items.push(value) })\n          return iteratorFor(items)\n        }\n\n        Headers.prototype.entries = function() {\n          var items = []\n          this.forEach(function(value, name) { items.push([name, value]) })\n          return iteratorFor(items)\n        }\n\n        if (support.iterable) {\n          Headers.prototype[Symbol.iterator] = Headers.prototype.entries\n        }\n\n        function consumed(body) {\n          if (body.bodyUsed) {\n            return Promise.reject(new TypeError('Already read'))\n          }\n          body.bodyUsed = true\n        }\n\n        function fileReaderReady(reader) {\n          return new Promise(function(resolve, reject) {\n            reader.onload = function() {\n              resolve(reader.result)\n            }\n            reader.onerror = function() {\n              reject(reader.error)\n            }\n          })\n        }\n\n        function readBlobAsArrayBuffer(blob) {\n          var reader = new FileReader()\n          var promise = fileReaderReady(reader)\n          reader.readAsArrayBuffer(blob)\n          return promise\n        }\n\n        function readBlobAsText(blob) {\n          var reader = new FileReader()\n          var promise = fileReaderReady(reader)\n          reader.readAsText(blob)\n          return promise\n        }\n\n        function readArrayBufferAsText(buf) {\n          var view = new Uint8Array(buf)\n          var chars = new Array(view.length)\n\n          for (var i = 0; i < view.length; i++) {\n            chars[i] = String.fromCharCode(view[i])\n          }\n          return chars.join('')\n        }\n\n        function bufferClone(buf) {\n          if (buf.slice) {\n            return buf.slice(0)\n          } else {\n            var view = new Uint8Array(buf.byteLength)\n            view.set(new Uint8Array(buf))\n            return view.buffer\n          }\n        }\n\n        function Body() {\n          this.bodyUsed = false\n\n          this._initBody = function(body) {\n            this._bodyInit = body\n            if (!body) {\n              this._bodyText = ''\n            } else if (typeof body === 'string') {\n              this._bodyText = body\n            } else if (support.blob && Blob.prototype.isPrototypeOf(body)) {\n              this._bodyBlob = body\n            } else if (support.formData && FormData.prototype.isPrototypeOf(body)) {\n              this._bodyFormData = body\n            } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n              this._bodyText = body.toString()\n            } else if (support.arrayBuffer && support.blob && isDataView(body)) {\n              this._bodyArrayBuffer = bufferClone(body.buffer)\n              // IE 10-11 can't handle a DataView body.\n              this._bodyInit = new Blob([this._bodyArrayBuffer])\n            } else if (support.arrayBuffer && (ArrayBuffer.prototype.isPrototypeOf(body) || isArrayBufferView(body))) {\n              this._bodyArrayBuffer = bufferClone(body)\n            } else {\n              throw new Error('unsupported BodyInit type')\n            }\n\n            if (!this.headers.get('content-type')) {\n              if (typeof body === 'string') {\n                this.headers.set('content-type', 'text/plain;charset=UTF-8')\n              } else if (this._bodyBlob && this._bodyBlob.type) {\n                this.headers.set('content-type', this._bodyBlob.type)\n              } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n                this.headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8')\n              }\n            }\n          }\n\n          if (support.blob) {\n            this.blob = function() {\n              var rejected = consumed(this)\n              if (rejected) {\n                return rejected\n              }\n\n              if (this._bodyBlob) {\n                return Promise.resolve(this._bodyBlob)\n              } else if (this._bodyArrayBuffer) {\n                return Promise.resolve(new Blob([this._bodyArrayBuffer]))\n              } else if (this._bodyFormData) {\n                throw new Error('could not read FormData body as blob')\n              } else {\n                return Promise.resolve(new Blob([this._bodyText]))\n              }\n            }\n\n            this.arrayBuffer = function() {\n              if (this._bodyArrayBuffer) {\n                return consumed(this) || Promise.resolve(this._bodyArrayBuffer)\n              } else {\n                return this.blob().then(readBlobAsArrayBuffer)\n              }\n            }\n          }\n\n          this.text = function() {\n            var rejected = consumed(this)\n            if (rejected) {\n              return rejected\n            }\n\n            if (this._bodyBlob) {\n              return readBlobAsText(this._bodyBlob)\n            } else if (this._bodyArrayBuffer) {\n              return Promise.resolve(readArrayBufferAsText(this._bodyArrayBuffer))\n            } else if (this._bodyFormData) {\n              throw new Error('could not read FormData body as text')\n            } else {\n              return Promise.resolve(this._bodyText)\n            }\n          }\n\n          if (support.formData) {\n            this.formData = function() {\n              return this.text().then(decode)\n            }\n          }\n\n          this.json = function() {\n            return this.text().then(JSON.parse)\n          }\n\n          return this\n        }\n\n        // HTTP methods whose capitalization should be normalized\n        var methods = ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'POST', 'PUT']\n\n        function normalizeMethod(method) {\n          var upcased = method.toUpperCase()\n          return (methods.indexOf(upcased) > -1) ? upcased : method\n        }\n\n        function Request(input, options) {\n          options = options || {}\n          var body = options.body\n\n          if (input instanceof Request) {\n            if (input.bodyUsed) {\n              throw new TypeError('Already read')\n            }\n            this.url = input.url\n            this.credentials = input.credentials\n            if (!options.headers) {\n              this.headers = new Headers(input.headers)\n            }\n            this.method = input.method\n            this.mode = input.mode\n            if (!body && input._bodyInit != null) {\n              body = input._bodyInit\n              input.bodyUsed = true\n            }\n          } else {\n            this.url = String(input)\n          }\n\n          this.credentials = options.credentials || this.credentials || 'omit'\n          if (options.headers || !this.headers) {\n            this.headers = new Headers(options.headers)\n          }\n          this.method = normalizeMethod(options.method || this.method || 'GET')\n          this.mode = options.mode || this.mode || null\n          this.referrer = null\n\n          if ((this.method === 'GET' || this.method === 'HEAD') && body) {\n            throw new TypeError('Body not allowed for GET or HEAD requests')\n          }\n          this._initBody(body)\n        }\n\n        Request.prototype.clone = function() {\n          return new Request(this, { body: this._bodyInit })\n        }\n\n        function decode(body) {\n          var form = new FormData()\n          body.trim().split('&').forEach(function(bytes) {\n            if (bytes) {\n              var split = bytes.split('=')\n              var name = split.shift().replace(/\\+/g, ' ')\n              var value = split.join('=').replace(/\\+/g, ' ')\n              form.append(decodeURIComponent(name), decodeURIComponent(value))\n            }\n          })\n          return form\n        }\n\n        function parseHeaders(rawHeaders) {\n          var headers = new Headers()\n          rawHeaders.split(/\\r?\\n/).forEach(function(line) {\n            var parts = line.split(':')\n            var key = parts.shift().trim()\n            if (key) {\n              var value = parts.join(':').trim()\n              headers.append(key, value)\n            }\n          })\n          return headers\n        }\n\n        Body.call(Request.prototype)\n\n        function Response(bodyInit, options) {\n          if (!options) {\n            options = {}\n          }\n\n          this.type = 'default'\n          this.status = 'status' in options ? options.status : 200\n          this.ok = this.status >= 200 && this.status < 300\n          this.statusText = 'statusText' in options ? options.statusText : 'OK'\n          this.headers = new Headers(options.headers)\n          this.url = options.url || ''\n          this._initBody(bodyInit)\n        }\n\n        Body.call(Response.prototype)\n\n        Response.prototype.clone = function() {\n          return new Response(this._bodyInit, {\n            status: this.status,\n            statusText: this.statusText,\n            headers: new Headers(this.headers),\n            url: this.url\n          })\n        }\n\n        Response.error = function() {\n          var response = new Response(null, {status: 0, statusText: ''})\n          response.type = 'error'\n          return response\n        }\n\n        var redirectStatuses = [301, 302, 303, 307, 308]\n\n        Response.redirect = function(url, status) {\n          if (redirectStatuses.indexOf(status) === -1) {\n            throw new RangeError('Invalid status code')\n          }\n\n          return new Response(null, {status: status, headers: {location: url}})\n        }\n\n        self.Headers = Headers\n        self.Request = Request\n        self.Response = Response\n\n        self.fetch = function(input, init) {\n          return new Promise(function(resolve, reject) {\n            var request = new Request(input, init)\n            var xhr = new XMLHttpRequest()\n\n            xhr.onload = function() {\n              var options = {\n                status: xhr.status,\n                statusText: xhr.statusText,\n                headers: parseHeaders(xhr.getAllResponseHeaders() || '')\n              }\n              options.url = 'responseURL' in xhr ? xhr.responseURL : options.headers.get('X-Request-URL')\n              var body = 'response' in xhr ? xhr.response : xhr.responseText\n              resolve(new Response(body, options))\n            }\n\n            xhr.onerror = function() {\n              reject(new TypeError('Network request failed'))\n            }\n\n            xhr.ontimeout = function() {\n              reject(new TypeError('Network request failed'))\n            }\n\n            xhr.open(request.method, request.url, true)\n\n            if (request.credentials === 'include') {\n              xhr.withCredentials = true\n            }\n\n            if ('responseType' in xhr && support.blob) {\n              xhr.responseType = 'blob'\n            }\n\n            request.headers.forEach(function(value, name) {\n              xhr.setRequestHeader(name, value)\n            })\n\n            xhr.send(typeof request._bodyInit === 'undefined' ? null : request._bodyInit)\n          })\n        }\n        self.fetch.polyfill = true\n      })(typeof self !== 'undefined' ? self : this);\n\n\n      return {\n        fetch: self.fetch,\n        Headers: self.Headers,\n        Request: self.Request,\n        Response: self.Response\n      };\n    }());\n  }\n\n  if (typeof define === 'function' && define.amd) {\n    define(function () {\n      return fetchPonyfill;\n    });\n  } else if (typeof exports === 'object') {\n    module.exports = fetchPonyfill;\n  } else {\n    self.fetchPonyfill = fetchPonyfill;\n  }\n}(typeof self === 'undefined' ? this : self));\n\n","// shim for using process in browser\nvar process = module.exports = {};\n\n// cached from whatever global is present so that test runners that stub it\n// don't break things.  But we need to wrap it in a try catch in case it is\n// wrapped in strict mode code which doesn't define any globals.  It's inside a\n// function because try/catches deoptimize in certain engines.\n\nvar cachedSetTimeout;\nvar cachedClearTimeout;\n\nfunction defaultSetTimout() {\n    throw new Error('setTimeout has not been defined');\n}\nfunction defaultClearTimeout () {\n    throw new Error('clearTimeout has not been defined');\n}\n(function () {\n    try {\n        if (typeof setTimeout === 'function') {\n            cachedSetTimeout = setTimeout;\n        } else {\n            cachedSetTimeout = defaultSetTimout;\n        }\n    } catch (e) {\n        cachedSetTimeout = defaultSetTimout;\n    }\n    try {\n        if (typeof clearTimeout === 'function') {\n            cachedClearTimeout = clearTimeout;\n        } else {\n            cachedClearTimeout = defaultClearTimeout;\n        }\n    } catch (e) {\n        cachedClearTimeout = defaultClearTimeout;\n    }\n} ())\nfunction runTimeout(fun) {\n    if (cachedSetTimeout === setTimeout) {\n        //normal enviroments in sane situations\n        return setTimeout(fun, 0);\n    }\n    // if setTimeout wasn't available but was latter defined\n    if ((cachedSetTimeout === defaultSetTimout || !cachedSetTimeout) && setTimeout) {\n        cachedSetTimeout = setTimeout;\n        return setTimeout(fun, 0);\n    }\n    try {\n        // when when somebody has screwed with setTimeout but no I.E. maddness\n        return cachedSetTimeout(fun, 0);\n    } catch(e){\n        try {\n            // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally\n            return cachedSetTimeout.call(null, fun, 0);\n        } catch(e){\n            // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error\n            return cachedSetTimeout.call(this, fun, 0);\n        }\n    }\n\n\n}\nfunction runClearTimeout(marker) {\n    if (cachedClearTimeout === clearTimeout) {\n        //normal enviroments in sane situations\n        return clearTimeout(marker);\n    }\n    // if clearTimeout wasn't available but was latter defined\n    if ((cachedClearTimeout === defaultClearTimeout || !cachedClearTimeout) && clearTimeout) {\n        cachedClearTimeout = clearTimeout;\n        return clearTimeout(marker);\n    }\n    try {\n        // when when somebody has screwed with setTimeout but no I.E. maddness\n        return cachedClearTimeout(marker);\n    } catch (e){\n        try {\n            // When we are in I.E. but the script has been evaled so I.E. doesn't  trust the global object when called normally\n            return cachedClearTimeout.call(null, marker);\n        } catch (e){\n            // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error.\n            // Some versions of I.E. have different rules for clearTimeout vs setTimeout\n            return cachedClearTimeout.call(this, marker);\n        }\n    }\n\n\n\n}\nvar queue = [];\nvar draining = false;\nvar currentQueue;\nvar queueIndex = -1;\n\nfunction cleanUpNextTick() {\n    if (!draining || !currentQueue) {\n        return;\n    }\n    draining = false;\n    if (currentQueue.length) {\n        queue = currentQueue.concat(queue);\n    } else {\n        queueIndex = -1;\n    }\n    if (queue.length) {\n        drainQueue();\n    }\n}\n\nfunction drainQueue() {\n    if (draining) {\n        return;\n    }\n    var timeout = runTimeout(cleanUpNextTick);\n    draining = true;\n\n    var len = queue.length;\n    while(len) {\n        currentQueue = queue;\n        queue = [];\n        while (++queueIndex < len) {\n            if (currentQueue) {\n                currentQueue[queueIndex].run();\n            }\n        }\n        queueIndex = -1;\n        len = queue.length;\n    }\n    currentQueue = null;\n    draining = false;\n    runClearTimeout(timeout);\n}\n\nprocess.nextTick = function (fun) {\n    var args = new Array(arguments.length - 1);\n    if (arguments.length > 1) {\n        for (var i = 1; i < arguments.length; i++) {\n            args[i - 1] = arguments[i];\n        }\n    }\n    queue.push(new Item(fun, args));\n    if (queue.length === 1 && !draining) {\n        runTimeout(drainQueue);\n    }\n};\n\n// v8 likes predictible objects\nfunction Item(fun, array) {\n    this.fun = fun;\n    this.array = array;\n}\nItem.prototype.run = function () {\n    this.fun.apply(null, this.array);\n};\nprocess.title = 'browser';\nprocess.browser = true;\nprocess.env = {};\nprocess.argv = [];\nprocess.version = ''; // empty string to avoid regexp issues\nprocess.versions = {};\n\nfunction noop() {}\n\nprocess.on = noop;\nprocess.addListener = noop;\nprocess.once = noop;\nprocess.off = noop;\nprocess.removeListener = noop;\nprocess.removeAllListeners = noop;\nprocess.emit = noop;\nprocess.prependListener = noop;\nprocess.prependOnceListener = noop;\n\nprocess.listeners = function (name) { return [] }\n\nprocess.binding = function (name) {\n    throw new Error('process.binding is not supported');\n};\n\nprocess.cwd = function () { return '/' };\nprocess.chdir = function (dir) {\n    throw new Error('process.chdir is not supported');\n};\nprocess.umask = function() { return 0; };\n","'use strict';\n\nvar replace = String.prototype.replace;\nvar percentTwenties = /%20/g;\n\nmodule.exports = {\n    'default': 'RFC3986',\n    formatters: {\n        RFC1738: function (value) {\n            return replace.call(value, percentTwenties, '+');\n        },\n        RFC3986: function (value) {\n            return value;\n        }\n    },\n    RFC1738: 'RFC1738',\n    RFC3986: 'RFC3986'\n};\n","'use strict';\n\nvar stringify = require('./stringify');\nvar parse = require('./parse');\nvar formats = require('./formats');\n\nmodule.exports = {\n    formats: formats,\n    parse: parse,\n    stringify: stringify\n};\n","'use strict';\n\nvar utils = require('./utils');\n\nvar has = Object.prototype.hasOwnProperty;\n\nvar defaults = {\n    allowDots: false,\n    allowPrototypes: false,\n    arrayLimit: 20,\n    decoder: utils.decode,\n    delimiter: '&',\n    depth: 5,\n    parameterLimit: 1000,\n    plainObjects: false,\n    strictNullHandling: false\n};\n\nvar parseValues = function parseQueryStringValues(str, options) {\n    var obj = {};\n    var cleanStr = options.ignoreQueryPrefix ? str.replace(/^\\?/, '') : str;\n    var limit = options.parameterLimit === Infinity ? undefined : options.parameterLimit;\n    var parts = cleanStr.split(options.delimiter, limit);\n\n    for (var i = 0; i < parts.length; ++i) {\n        var part = parts[i];\n\n        var bracketEqualsPos = part.indexOf(']=');\n        var pos = bracketEqualsPos === -1 ? part.indexOf('=') : bracketEqualsPos + 1;\n\n        var key, val;\n        if (pos === -1) {\n            key = options.decoder(part, defaults.decoder);\n            val = options.strictNullHandling ? null : '';\n        } else {\n            key = options.decoder(part.slice(0, pos), defaults.decoder);\n            val = options.decoder(part.slice(pos + 1), defaults.decoder);\n        }\n        if (has.call(obj, key)) {\n            obj[key] = [].concat(obj[key]).concat(val);\n        } else {\n            obj[key] = val;\n        }\n    }\n\n    return obj;\n};\n\nvar parseObject = function (chain, val, options) {\n    var leaf = val;\n\n    for (var i = chain.length - 1; i >= 0; --i) {\n        var obj;\n        var root = chain[i];\n\n        if (root === '[]') {\n            obj = [];\n            obj = obj.concat(leaf);\n        } else {\n            obj = options.plainObjects ? Object.create(null) : {};\n            var cleanRoot = root.charAt(0) === '[' && root.charAt(root.length - 1) === ']' ? root.slice(1, -1) : root;\n            var index = parseInt(cleanRoot, 10);\n            if (\n                !isNaN(index)\n                && root !== cleanRoot\n                && String(index) === cleanRoot\n                && index >= 0\n                && (options.parseArrays && index <= options.arrayLimit)\n            ) {\n                obj = [];\n                obj[index] = leaf;\n            } else {\n                obj[cleanRoot] = leaf;\n            }\n        }\n\n        leaf = obj;\n    }\n\n    return leaf;\n};\n\nvar parseKeys = function parseQueryStringKeys(givenKey, val, options) {\n    if (!givenKey) {\n        return;\n    }\n\n    // Transform dot notation to bracket notation\n    var key = options.allowDots ? givenKey.replace(/\\.([^.[]+)/g, '[$1]') : givenKey;\n\n    // The regex chunks\n\n    var brackets = /(\\[[^[\\]]*])/;\n    var child = /(\\[[^[\\]]*])/g;\n\n    // Get the parent\n\n    var segment = brackets.exec(key);\n    var parent = segment ? key.slice(0, segment.index) : key;\n\n    // Stash the parent if it exists\n\n    var keys = [];\n    if (parent) {\n        // If we aren't using plain objects, optionally prefix keys\n        // that would overwrite object prototype properties\n        if (!options.plainObjects && has.call(Object.prototype, parent)) {\n            if (!options.allowPrototypes) {\n                return;\n            }\n        }\n\n        keys.push(parent);\n    }\n\n    // Loop through children appending to the array until we hit depth\n\n    var i = 0;\n    while ((segment = child.exec(key)) !== null && i < options.depth) {\n        i += 1;\n        if (!options.plainObjects && has.call(Object.prototype, segment[1].slice(1, -1))) {\n            if (!options.allowPrototypes) {\n                return;\n            }\n        }\n        keys.push(segment[1]);\n    }\n\n    // If there's a remainder, just add whatever is left\n\n    if (segment) {\n        keys.push('[' + key.slice(segment.index) + ']');\n    }\n\n    return parseObject(keys, val, options);\n};\n\nmodule.exports = function (str, opts) {\n    var options = opts ? utils.assign({}, opts) : {};\n\n    if (options.decoder !== null && options.decoder !== undefined && typeof options.decoder !== 'function') {\n        throw new TypeError('Decoder has to be a function.');\n    }\n\n    options.ignoreQueryPrefix = options.ignoreQueryPrefix === true;\n    options.delimiter = typeof options.delimiter === 'string' || utils.isRegExp(options.delimiter) ? options.delimiter : defaults.delimiter;\n    options.depth = typeof options.depth === 'number' ? options.depth : defaults.depth;\n    options.arrayLimit = typeof options.arrayLimit === 'number' ? options.arrayLimit : defaults.arrayLimit;\n    options.parseArrays = options.parseArrays !== false;\n    options.decoder = typeof options.decoder === 'function' ? options.decoder : defaults.decoder;\n    options.allowDots = typeof options.allowDots === 'boolean' ? options.allowDots : defaults.allowDots;\n    options.plainObjects = typeof options.plainObjects === 'boolean' ? options.plainObjects : defaults.plainObjects;\n    options.allowPrototypes = typeof options.allowPrototypes === 'boolean' ? options.allowPrototypes : defaults.allowPrototypes;\n    options.parameterLimit = typeof options.parameterLimit === 'number' ? options.parameterLimit : defaults.parameterLimit;\n    options.strictNullHandling = typeof options.strictNullHandling === 'boolean' ? options.strictNullHandling : defaults.strictNullHandling;\n\n    if (str === '' || str === null || typeof str === 'undefined') {\n        return options.plainObjects ? Object.create(null) : {};\n    }\n\n    var tempObj = typeof str === 'string' ? parseValues(str, options) : str;\n    var obj = options.plainObjects ? Object.create(null) : {};\n\n    // Iterate over the keys and setup the new object\n\n    var keys = Object.keys(tempObj);\n    for (var i = 0; i < keys.length; ++i) {\n        var key = keys[i];\n        var newObj = parseKeys(key, tempObj[key], options);\n        obj = utils.merge(obj, newObj, options);\n    }\n\n    return utils.compact(obj);\n};\n","'use strict';\n\nvar utils = require('./utils');\nvar formats = require('./formats');\n\nvar arrayPrefixGenerators = {\n    brackets: function brackets(prefix) { // eslint-disable-line func-name-matching\n        return prefix + '[]';\n    },\n    indices: function indices(prefix, key) { // eslint-disable-line func-name-matching\n        return prefix + '[' + key + ']';\n    },\n    repeat: function repeat(prefix) { // eslint-disable-line func-name-matching\n        return prefix;\n    }\n};\n\nvar toISO = Date.prototype.toISOString;\n\nvar defaults = {\n    delimiter: '&',\n    encode: true,\n    encoder: utils.encode,\n    encodeValuesOnly: false,\n    serializeDate: function serializeDate(date) { // eslint-disable-line func-name-matching\n        return toISO.call(date);\n    },\n    skipNulls: false,\n    strictNullHandling: false\n};\n\nvar stringify = function stringify( // eslint-disable-line func-name-matching\n    object,\n    prefix,\n    generateArrayPrefix,\n    strictNullHandling,\n    skipNulls,\n    encoder,\n    filter,\n    sort,\n    allowDots,\n    serializeDate,\n    formatter,\n    encodeValuesOnly\n) {\n    var obj = object;\n    if (typeof filter === 'function') {\n        obj = filter(prefix, obj);\n    } else if (obj instanceof Date) {\n        obj = serializeDate(obj);\n    } else if (obj === null) {\n        if (strictNullHandling) {\n            return encoder && !encodeValuesOnly ? encoder(prefix, defaults.encoder) : prefix;\n        }\n\n        obj = '';\n    }\n\n    if (typeof obj === 'string' || typeof obj === 'number' || typeof obj === 'boolean' || utils.isBuffer(obj)) {\n        if (encoder) {\n            var keyValue = encodeValuesOnly ? prefix : encoder(prefix, defaults.encoder);\n            return [formatter(keyValue) + '=' + formatter(encoder(obj, defaults.encoder))];\n        }\n        return [formatter(prefix) + '=' + formatter(String(obj))];\n    }\n\n    var values = [];\n\n    if (typeof obj === 'undefined') {\n        return values;\n    }\n\n    var objKeys;\n    if (Array.isArray(filter)) {\n        objKeys = filter;\n    } else {\n        var keys = Object.keys(obj);\n        objKeys = sort ? keys.sort(sort) : keys;\n    }\n\n    for (var i = 0; i < objKeys.length; ++i) {\n        var key = objKeys[i];\n\n        if (skipNulls && obj[key] === null) {\n            continue;\n        }\n\n        if (Array.isArray(obj)) {\n            values = values.concat(stringify(\n                obj[key],\n                generateArrayPrefix(prefix, key),\n                generateArrayPrefix,\n                strictNullHandling,\n                skipNulls,\n                encoder,\n                filter,\n                sort,\n                allowDots,\n                serializeDate,\n                formatter,\n                encodeValuesOnly\n            ));\n        } else {\n            values = values.concat(stringify(\n                obj[key],\n                prefix + (allowDots ? '.' + key : '[' + key + ']'),\n                generateArrayPrefix,\n                strictNullHandling,\n                skipNulls,\n                encoder,\n                filter,\n                sort,\n                allowDots,\n                serializeDate,\n                formatter,\n                encodeValuesOnly\n            ));\n        }\n    }\n\n    return values;\n};\n\nmodule.exports = function (object, opts) {\n    var obj = object;\n    var options = opts ? utils.assign({}, opts) : {};\n\n    if (options.encoder !== null && options.encoder !== undefined && typeof options.encoder !== 'function') {\n        throw new TypeError('Encoder has to be a function.');\n    }\n\n    var delimiter = typeof options.delimiter === 'undefined' ? defaults.delimiter : options.delimiter;\n    var strictNullHandling = typeof options.strictNullHandling === 'boolean' ? options.strictNullHandling : defaults.strictNullHandling;\n    var skipNulls = typeof options.skipNulls === 'boolean' ? options.skipNulls : defaults.skipNulls;\n    var encode = typeof options.encode === 'boolean' ? options.encode : defaults.encode;\n    var encoder = typeof options.encoder === 'function' ? options.encoder : defaults.encoder;\n    var sort = typeof options.sort === 'function' ? options.sort : null;\n    var allowDots = typeof options.allowDots === 'undefined' ? false : options.allowDots;\n    var serializeDate = typeof options.serializeDate === 'function' ? options.serializeDate : defaults.serializeDate;\n    var encodeValuesOnly = typeof options.encodeValuesOnly === 'boolean' ? options.encodeValuesOnly : defaults.encodeValuesOnly;\n    if (typeof options.format === 'undefined') {\n        options.format = formats['default'];\n    } else if (!Object.prototype.hasOwnProperty.call(formats.formatters, options.format)) {\n        throw new TypeError('Unknown format option provided.');\n    }\n    var formatter = formats.formatters[options.format];\n    var objKeys;\n    var filter;\n\n    if (typeof options.filter === 'function') {\n        filter = options.filter;\n        obj = filter('', obj);\n    } else if (Array.isArray(options.filter)) {\n        filter = options.filter;\n        objKeys = filter;\n    }\n\n    var keys = [];\n\n    if (typeof obj !== 'object' || obj === null) {\n        return '';\n    }\n\n    var arrayFormat;\n    if (options.arrayFormat in arrayPrefixGenerators) {\n        arrayFormat = options.arrayFormat;\n    } else if ('indices' in options) {\n        arrayFormat = options.indices ? 'indices' : 'repeat';\n    } else {\n        arrayFormat = 'indices';\n    }\n\n    var generateArrayPrefix = arrayPrefixGenerators[arrayFormat];\n\n    if (!objKeys) {\n        objKeys = Object.keys(obj);\n    }\n\n    if (sort) {\n        objKeys.sort(sort);\n    }\n\n    for (var i = 0; i < objKeys.length; ++i) {\n        var key = objKeys[i];\n\n        if (skipNulls && obj[key] === null) {\n            continue;\n        }\n\n        keys = keys.concat(stringify(\n            obj[key],\n            key,\n            generateArrayPrefix,\n            strictNullHandling,\n            skipNulls,\n            encode ? encoder : null,\n            filter,\n            sort,\n            allowDots,\n            serializeDate,\n            formatter,\n            encodeValuesOnly\n        ));\n    }\n\n    var joined = keys.join(delimiter);\n    var prefix = options.addQueryPrefix === true ? '?' : '';\n\n    return joined.length > 0 ? prefix + joined : '';\n};\n","'use strict';\n\nvar has = Object.prototype.hasOwnProperty;\n\nvar hexTable = (function () {\n    var array = [];\n    for (var i = 0; i < 256; ++i) {\n        array.push('%' + ((i < 16 ? '0' : '') + i.toString(16)).toUpperCase());\n    }\n\n    return array;\n}());\n\nvar compactQueue = function compactQueue(queue) {\n    var obj;\n\n    while (queue.length) {\n        var item = queue.pop();\n        obj = item.obj[item.prop];\n\n        if (Array.isArray(obj)) {\n            var compacted = [];\n\n            for (var j = 0; j < obj.length; ++j) {\n                if (typeof obj[j] !== 'undefined') {\n                    compacted.push(obj[j]);\n                }\n            }\n\n            item.obj[item.prop] = compacted;\n        }\n    }\n\n    return obj;\n};\n\nexports.arrayToObject = function arrayToObject(source, options) {\n    var obj = options && options.plainObjects ? Object.create(null) : {};\n    for (var i = 0; i < source.length; ++i) {\n        if (typeof source[i] !== 'undefined') {\n            obj[i] = source[i];\n        }\n    }\n\n    return obj;\n};\n\nexports.merge = function merge(target, source, options) {\n    if (!source) {\n        return target;\n    }\n\n    if (typeof source !== 'object') {\n        if (Array.isArray(target)) {\n            target.push(source);\n        } else if (typeof target === 'object') {\n            if (options.plainObjects || options.allowPrototypes || !has.call(Object.prototype, source)) {\n                target[source] = true;\n            }\n        } else {\n            return [target, source];\n        }\n\n        return target;\n    }\n\n    if (typeof target !== 'object') {\n        return [target].concat(source);\n    }\n\n    var mergeTarget = target;\n    if (Array.isArray(target) && !Array.isArray(source)) {\n        mergeTarget = exports.arrayToObject(target, options);\n    }\n\n    if (Array.isArray(target) && Array.isArray(source)) {\n        source.forEach(function (item, i) {\n            if (has.call(target, i)) {\n                if (target[i] && typeof target[i] === 'object') {\n                    target[i] = exports.merge(target[i], item, options);\n                } else {\n                    target.push(item);\n                }\n            } else {\n                target[i] = item;\n            }\n        });\n        return target;\n    }\n\n    return Object.keys(source).reduce(function (acc, key) {\n        var value = source[key];\n\n        if (has.call(acc, key)) {\n            acc[key] = exports.merge(acc[key], value, options);\n        } else {\n            acc[key] = value;\n        }\n        return acc;\n    }, mergeTarget);\n};\n\nexports.assign = function assignSingleSource(target, source) {\n    return Object.keys(source).reduce(function (acc, key) {\n        acc[key] = source[key];\n        return acc;\n    }, target);\n};\n\nexports.decode = function (str) {\n    try {\n        return decodeURIComponent(str.replace(/\\+/g, ' '));\n    } catch (e) {\n        return str;\n    }\n};\n\nexports.encode = function encode(str) {\n    // This code was originally written by Brian White (mscdex) for the io.js core querystring library.\n    // It has been adapted here for stricter adherence to RFC 3986\n    if (str.length === 0) {\n        return str;\n    }\n\n    var string = typeof str === 'string' ? str : String(str);\n\n    var out = '';\n    for (var i = 0; i < string.length; ++i) {\n        var c = string.charCodeAt(i);\n\n        if (\n            c === 0x2D // -\n            || c === 0x2E // .\n            || c === 0x5F // _\n            || c === 0x7E // ~\n            || (c >= 0x30 && c <= 0x39) // 0-9\n            || (c >= 0x41 && c <= 0x5A) // a-z\n            || (c >= 0x61 && c <= 0x7A) // A-Z\n        ) {\n            out += string.charAt(i);\n            continue;\n        }\n\n        if (c < 0x80) {\n            out = out + hexTable[c];\n            continue;\n        }\n\n        if (c < 0x800) {\n            out = out + (hexTable[0xC0 | (c >> 6)] + hexTable[0x80 | (c & 0x3F)]);\n            continue;\n        }\n\n        if (c < 0xD800 || c >= 0xE000) {\n            out = out + (hexTable[0xE0 | (c >> 12)] + hexTable[0x80 | ((c >> 6) & 0x3F)] + hexTable[0x80 | (c & 0x3F)]);\n            continue;\n        }\n\n        i += 1;\n        c = 0x10000 + (((c & 0x3FF) << 10) | (string.charCodeAt(i) & 0x3FF));\n        out += hexTable[0xF0 | (c >> 18)]\n            + hexTable[0x80 | ((c >> 12) & 0x3F)]\n            + hexTable[0x80 | ((c >> 6) & 0x3F)]\n            + hexTable[0x80 | (c & 0x3F)];\n    }\n\n    return out;\n};\n\nexports.compact = function compact(value) {\n    var queue = [{ obj: { o: value }, prop: 'o' }];\n    var refs = [];\n\n    for (var i = 0; i < queue.length; ++i) {\n        var item = queue[i];\n        var obj = item.obj[item.prop];\n\n        var keys = Object.keys(obj);\n        for (var j = 0; j < keys.length; ++j) {\n            var key = keys[j];\n            var val = obj[key];\n            if (typeof val === 'object' && val !== null && refs.indexOf(val) === -1) {\n                queue.push({ obj: obj, prop: key });\n                refs.push(val);\n            }\n        }\n    }\n\n    return compactQueue(queue);\n};\n\nexports.isRegExp = function isRegExp(obj) {\n    return Object.prototype.toString.call(obj) === '[object RegExp]';\n};\n\nexports.isBuffer = function isBuffer(obj) {\n    if (obj === null || typeof obj === 'undefined') {\n        return false;\n    }\n\n    return !!(obj.constructor && obj.constructor.isBuffer && obj.constructor.isBuffer(obj));\n};\n"]} +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","ccxt.browser.js","ccxt.js","js/_1broker.js","js/_1btcxe.js","js/acx.js","js/allcoin.js","js/anxpro.js","js/base/Exchange.js","js/base/Market.js","js/base/errors.js","js/base/functions.js","js/base/functions/crypto.js","js/base/functions/encode.js","js/base/functions/generic.js","js/base/functions/misc.js","js/base/functions/number.js","js/base/functions/platform.js","js/base/functions/string.js","js/base/functions/throttle.js","js/base/functions/time.js","js/base/functions/type.js","js/bibox.js","js/binance.js","js/bit2c.js","js/bitbay.js","js/bitcoincoid.js","js/bitfinex.js","js/bitfinex2.js","js/bitflyer.js","js/bithumb.js","js/bitlish.js","js/bitmarket.js","js/bitmex.js","js/bitso.js","js/bitstamp.js","js/bitstamp1.js","js/bittrex.js","js/bl3p.js","js/bleutrade.js","js/braziliex.js","js/btcbox.js","js/btcchina.js","js/btcexchange.js","js/btcmarkets.js","js/btctradeua.js","js/btcturk.js","js/btcx.js","js/bter.js","js/bxinth.js","js/ccex.js","js/cex.js","js/chbtc.js","js/chilebit.js","js/coincheck.js","js/coinexchange.js","js/coinfloor.js","js/coingi.js","js/coinmarketcap.js","js/coinmate.js","js/coinsecure.js","js/coinspot.js","js/cryptopia.js","js/dsx.js","js/exmo.js","js/flowbtc.js","js/foxbit.js","js/fybse.js","js/fybsg.js","js/gatecoin.js","js/gateio.js","js/gdax.js","js/gemini.js","js/getbtc.js","js/hitbtc.js","js/hitbtc2.js","js/huobi.js","js/huobicny.js","js/huobipro.js","js/independentreserve.js","js/itbit.js","js/jubi.js","js/kraken.js","js/kucoin.js","js/kuna.js","js/lakebtc.js","js/liqui.js","js/livecoin.js","js/luno.js","js/lykke.js","js/mercado.js","js/mixcoins.js","js/nova.js","js/okcoincny.js","js/okcoinusd.js","js/okex.js","js/paymium.js","js/poloniex.js","js/qryptos.js","js/quadrigacx.js","js/quoinex.js","js/southxchange.js","js/surbitcoin.js","js/therock.js","js/tidex.js","js/urdubit.js","js/vaultoro.js","js/vbtc.js","js/virwox.js","js/wex.js","js/xbtce.js","js/yobit.js","js/yunbi.js","js/zaif.js","js/zb.js","node_modules/crypto-js/aes.js","node_modules/crypto-js/cipher-core.js","node_modules/crypto-js/core.js","node_modules/crypto-js/enc-base64.js","node_modules/crypto-js/enc-utf16.js","node_modules/crypto-js/evpkdf.js","node_modules/crypto-js/format-hex.js","node_modules/crypto-js/hmac.js","node_modules/crypto-js/index.js","node_modules/crypto-js/lib-typedarrays.js","node_modules/crypto-js/md5.js","node_modules/crypto-js/mode-cfb.js","node_modules/crypto-js/mode-ctr-gladman.js","node_modules/crypto-js/mode-ctr.js","node_modules/crypto-js/mode-ecb.js","node_modules/crypto-js/mode-ofb.js","node_modules/crypto-js/pad-ansix923.js","node_modules/crypto-js/pad-iso10126.js","node_modules/crypto-js/pad-iso97971.js","node_modules/crypto-js/pad-nopadding.js","node_modules/crypto-js/pad-zeropadding.js","node_modules/crypto-js/pbkdf2.js","node_modules/crypto-js/rabbit-legacy.js","node_modules/crypto-js/rabbit.js","node_modules/crypto-js/rc4.js","node_modules/crypto-js/ripemd160.js","node_modules/crypto-js/sha1.js","node_modules/crypto-js/sha224.js","node_modules/crypto-js/sha256.js","node_modules/crypto-js/sha3.js","node_modules/crypto-js/sha384.js","node_modules/crypto-js/sha512.js","node_modules/crypto-js/tripledes.js","node_modules/crypto-js/x64-core.js","node_modules/fetch-ponyfill/build/fetch-browser.js","node_modules/process/browser.js","node_modules/qs/lib/formats.js","node_modules/qs/lib/index.js","node_modules/qs/lib/parse.js","node_modules/qs/lib/stringify.js","node_modules/qs/lib/utils.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;;ACLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/PA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACrOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;ACh1BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACXA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzgBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC10BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1UA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnsBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzaA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClWA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrXA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7sBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7MA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9cA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9XA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1SA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5QA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3dA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5MA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxUA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9mBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9SA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjaA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACriBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACz2BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7iCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/dA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3OA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5xBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1mBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtqBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACljBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/WA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvSA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvoBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClyBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9PA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7RA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3MA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxWA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/2BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvvBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3QA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1QA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClUA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClUA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjwBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/SA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACXA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","/*  A entry point for the browser bundle version. This gets compiled by:\n        \n        browserify --debug ./ccxt.browser.js > ./build/ccxt.browser.js\n */\n\nwindow.ccxt = require ('./ccxt')","\"use strict\";\n\n/*\n\nMIT License\n\nCopyright (c) 2017 Igor Kroitor\n\nPermission is hereby granted, free of charge, to any person obtaining a copy\nof this software and associated documentation files (the \"Software\"), to deal\nin the Software without restriction, including without limitation the rights\nto use, copy, modify, merge, publish, distribute, sublicense, and/or sell\ncopies of the Software, and to permit persons to whom the Software is\nfurnished to do so, subject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\nFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\nAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\nLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\nOUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\nSOFTWARE.\n\n*/\n\n\"use strict\";\n\n//-----------------------------------------------------------------------------\n\nconst Exchange  = require ('./js/base/Exchange')\n    , functions = require ('./js/base/functions')\n    , errors    = require ('./js/base/errors')\n\n//-----------------------------------------------------------------------------\n// this is updated by vss.js when building\n\nconst version = '1.10.760'\n\nExchange.ccxtVersion = version\n\n//-----------------------------------------------------------------------------\n\nconst exchanges = {\n    '_1broker':                require ('./js/_1broker.js'),\n    '_1btcxe':                 require ('./js/_1btcxe.js'),\n    'acx':                     require ('./js/acx.js'),\n    'allcoin':                 require ('./js/allcoin.js'),\n    'anxpro':                  require ('./js/anxpro.js'),\n    'bibox':                   require ('./js/bibox.js'),\n    'binance':                 require ('./js/binance.js'),\n    'bit2c':                   require ('./js/bit2c.js'),\n    'bitbay':                  require ('./js/bitbay.js'),\n    'bitcoincoid':             require ('./js/bitcoincoid.js'),\n    'bitfinex':                require ('./js/bitfinex.js'),\n    'bitfinex2':               require ('./js/bitfinex2.js'),\n    'bitflyer':                require ('./js/bitflyer.js'),\n    'bithumb':                 require ('./js/bithumb.js'),\n    'bitlish':                 require ('./js/bitlish.js'),\n    'bitmarket':               require ('./js/bitmarket.js'),\n    'bitmex':                  require ('./js/bitmex.js'),\n    'bitso':                   require ('./js/bitso.js'),\n    'bitstamp':                require ('./js/bitstamp.js'),\n    'bitstamp1':               require ('./js/bitstamp1.js'),\n    'bittrex':                 require ('./js/bittrex.js'),\n    'bl3p':                    require ('./js/bl3p.js'),\n    'bleutrade':               require ('./js/bleutrade.js'),\n    'braziliex':               require ('./js/braziliex.js'),\n    'btcbox':                  require ('./js/btcbox.js'),\n    'btcchina':                require ('./js/btcchina.js'),\n    'btcexchange':             require ('./js/btcexchange.js'),\n    'btcmarkets':              require ('./js/btcmarkets.js'),\n    'btctradeua':              require ('./js/btctradeua.js'),\n    'btcturk':                 require ('./js/btcturk.js'),\n    'btcx':                    require ('./js/btcx.js'),\n    'bter':                    require ('./js/bter.js'),\n    'bxinth':                  require ('./js/bxinth.js'),\n    'ccex':                    require ('./js/ccex.js'),\n    'cex':                     require ('./js/cex.js'),\n    'chbtc':                   require ('./js/chbtc.js'),\n    'chilebit':                require ('./js/chilebit.js'),\n    'coincheck':               require ('./js/coincheck.js'),\n    'coinexchange':            require ('./js/coinexchange.js'),\n    'coinfloor':               require ('./js/coinfloor.js'),\n    'coingi':                  require ('./js/coingi.js'),\n    'coinmarketcap':           require ('./js/coinmarketcap.js'),\n    'coinmate':                require ('./js/coinmate.js'),\n    'coinsecure':              require ('./js/coinsecure.js'),\n    'coinspot':                require ('./js/coinspot.js'),\n    'cryptopia':               require ('./js/cryptopia.js'),\n    'dsx':                     require ('./js/dsx.js'),\n    'exmo':                    require ('./js/exmo.js'),\n    'flowbtc':                 require ('./js/flowbtc.js'),\n    'foxbit':                  require ('./js/foxbit.js'),\n    'fybse':                   require ('./js/fybse.js'),\n    'fybsg':                   require ('./js/fybsg.js'),\n    'gatecoin':                require ('./js/gatecoin.js'),\n    'gateio':                  require ('./js/gateio.js'),\n    'gdax':                    require ('./js/gdax.js'),\n    'gemini':                  require ('./js/gemini.js'),\n    'getbtc':                  require ('./js/getbtc.js'),\n    'hitbtc':                  require ('./js/hitbtc.js'),\n    'hitbtc2':                 require ('./js/hitbtc2.js'),\n    'huobi':                   require ('./js/huobi.js'),\n    'huobicny':                require ('./js/huobicny.js'),\n    'huobipro':                require ('./js/huobipro.js'),\n    'independentreserve':      require ('./js/independentreserve.js'),\n    'itbit':                   require ('./js/itbit.js'),\n    'jubi':                    require ('./js/jubi.js'),\n    'kraken':                  require ('./js/kraken.js'),\n    'kucoin':                  require ('./js/kucoin.js'),\n    'kuna':                    require ('./js/kuna.js'),\n    'lakebtc':                 require ('./js/lakebtc.js'),\n    'liqui':                   require ('./js/liqui.js'),\n    'livecoin':                require ('./js/livecoin.js'),\n    'luno':                    require ('./js/luno.js'),\n    'lykke':                   require ('./js/lykke.js'),\n    'mercado':                 require ('./js/mercado.js'),\n    'mixcoins':                require ('./js/mixcoins.js'),\n    'nova':                    require ('./js/nova.js'),\n    'okcoincny':               require ('./js/okcoincny.js'),\n    'okcoinusd':               require ('./js/okcoinusd.js'),\n    'okex':                    require ('./js/okex.js'),\n    'paymium':                 require ('./js/paymium.js'),\n    'poloniex':                require ('./js/poloniex.js'),\n    'qryptos':                 require ('./js/qryptos.js'),\n    'quadrigacx':              require ('./js/quadrigacx.js'),\n    'quoinex':                 require ('./js/quoinex.js'),\n    'southxchange':            require ('./js/southxchange.js'),\n    'surbitcoin':              require ('./js/surbitcoin.js'),\n    'therock':                 require ('./js/therock.js'),\n    'tidex':                   require ('./js/tidex.js'),\n    'urdubit':                 require ('./js/urdubit.js'),\n    'vaultoro':                require ('./js/vaultoro.js'),\n    'vbtc':                    require ('./js/vbtc.js'),\n    'virwox':                  require ('./js/virwox.js'),\n    'wex':                     require ('./js/wex.js'),\n    'xbtce':                   require ('./js/xbtce.js'),\n    'yobit':                   require ('./js/yobit.js'),\n    'yunbi':                   require ('./js/yunbi.js'),\n    'zaif':                    require ('./js/zaif.js'),\n    'zb':                      require ('./js/zb.js'),    \n}\n\n//-----------------------------------------------------------------------------\n\nmodule.exports = Object.assign ({ version, Exchange, exchanges: Object.keys (exchanges) }, exchanges, functions, errors)\n\n//-----------------------------------------------------------------------------\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class _1broker extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': '_1broker',\n            'name': '1Broker',\n            'countries': 'US',\n            'rateLimit': 1500,\n            'version': 'v2',\n            'has': {\n                'publicAPI': false,\n                'CORS': true,\n                'fetchTrades': false,\n                'fetchOHLCV': true,\n            },\n            'timeframes': {\n                '1m': '60',\n                '15m': '900',\n                '1h': '3600',\n                '1d': '86400',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766021-420bd9fc-5ecb-11e7-8ed6-56d0081efed2.jpg',\n                'api': 'https://1broker.com/api',\n                'www': 'https://1broker.com',\n                'doc': 'https://1broker.com/?c=en/content/api-documentation',\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': false,\n            },\n            'api': {\n                'private': {\n                    'get': [\n                        'market/bars',\n                        'market/categories',\n                        'market/details',\n                        'market/list',\n                        'market/quotes',\n                        'market/ticks',\n                        'order/cancel',\n                        'order/create',\n                        'order/open',\n                        'position/close',\n                        'position/close_cancel',\n                        'position/edit',\n                        'position/history',\n                        'position/open',\n                        'position/shared/get',\n                        'social/profile_statistics',\n                        'social/profile_trades',\n                        'user/bitcoin_deposit_address',\n                        'user/details',\n                        'user/overview',\n                        'user/quota_status',\n                        'user/transaction_log',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchCategories () {\n        let response = await this.privateGetMarketCategories ();\n        // they return an empty string among their categories, wtf?\n        let categories = response['response'];\n        let result = [];\n        for (let i = 0; i < categories.length; i++) {\n            if (categories[i])\n                result.push (categories[i]);\n        }\n        return result;\n    }\n\n    async fetchMarkets () {\n        let this_ = this; // workaround for Babel bug (not passing `this` to _recursive() call)\n        let categories = await this.fetchCategories ();\n        let result = [];\n        for (let c = 0; c < categories.length; c++) {\n            let category = categories[c];\n            let markets = await this_.privateGetMarketList ({\n                'category': category.toLowerCase (),\n            });\n            for (let p = 0; p < markets['response'].length; p++) {\n                let market = markets['response'][p];\n                let id = market['symbol'];\n                let symbol = undefined;\n                let base = undefined;\n                let quote = undefined;\n                if ((category == 'FOREX') || (category == 'CRYPTO')) {\n                    symbol = market['name'];\n                    let parts = symbol.split ('/');\n                    base = parts[0];\n                    quote = parts[1];\n                } else {\n                    base = id;\n                    quote = 'USD';\n                    symbol = base + '/' + quote;\n                }\n                base = this_.commonCurrencyCode (base);\n                quote = this_.commonCurrencyCode (quote);\n                result.push ({\n                    'id': id,\n                    'symbol': symbol,\n                    'base': base,\n                    'quote': quote,\n                    'info': market,\n                });\n            }\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balance = await this.privateGetUserOverview ();\n        let response = balance['response'];\n        let result = {\n            'info': response,\n        };\n        let currencies = Object.keys (this.currencies);\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            result[currency] = this.account ();\n        }\n        let total = parseFloat (response['balance']);\n        result['BTC']['free'] = total;\n        result['BTC']['total'] = total;\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetMarketQuotes (this.extend ({\n            'symbols': this.marketId (symbol),\n        }, params));\n        let orderbook = response['response'][0];\n        let timestamp = this.parse8601 (orderbook['updated']);\n        let bidPrice = parseFloat (orderbook['bid']);\n        let askPrice = parseFloat (orderbook['ask']);\n        let bid = [ bidPrice, undefined ];\n        let ask = [ askPrice, undefined ];\n        return {\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'bids': [ bid ],\n            'asks': [ ask ],\n        };\n    }\n\n    async fetchTrades (symbol) {\n        throw new ExchangeError (this.id + ' fetchTrades () method not implemented yet');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let result = await this.privateGetMarketBars (this.extend ({\n            'symbol': this.marketId (symbol),\n            'resolution': 60,\n            'limit': 1,\n        }, params));\n        let orderbook = await this.fetchOrderBook (symbol);\n        let ticker = result['response'][0];\n        let timestamp = this.parse8601 (ticker['date']);\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['h']),\n            'low': parseFloat (ticker['l']),\n            'bid': orderbook['bids'][0][0],\n            'ask': orderbook['asks'][0][0],\n            'vwap': undefined,\n            'open': parseFloat (ticker['o']),\n            'close': parseFloat (ticker['c']),\n            'first': undefined,\n            'last': undefined,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            this.parse8601 (ohlcv['date']),\n            parseFloat (ohlcv['o']),\n            parseFloat (ohlcv['h']),\n            parseFloat (ohlcv['l']),\n            parseFloat (ohlcv['c']),\n            undefined,\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n            'resolution': this.timeframes[timeframe],\n        };\n        if (since)\n            request['date_start'] = this.iso8601 (since); // they also support date_end\n        if (limit)\n            request['limit'] = limit;\n        let result = await this.privateGetMarketBars (this.extend (request, params));\n        return this.parseOHLCVs (result['response'], market, timeframe, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'symbol': this.marketId (symbol),\n            'margin': amount,\n            'direction': (side == 'sell') ? 'short' : 'long',\n            'leverage': 1,\n            'type': side,\n        };\n        if (type == 'limit')\n            order['price'] = price;\n        else\n            order['type'] += '_market';\n        let result = await this.privateGetOrderCreate (this.extend (order, params));\n        return {\n            'info': result,\n            'id': result['response']['order_id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostOrderCancel ({ 'order_id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        this.checkRequiredCredentials ();\n        let url = this.urls['api'] + '/' + this.version + '/' + path + '.php';\n        let query = this.extend ({ 'token': this.apiKey }, params);\n        url += '?' + this.urlencode (query);\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('warning' in response)\n            if (response['warning'])\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        if ('error' in response)\n            if (response['error'])\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class _1btcxe extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': '_1btcxe',\n            'name': '1BTCXE',\n            'countries': 'PA', // Panama\n            'comment': 'Crypto Capital API',\n            'has': {\n                'CORS': true,\n                'fetchTickers': true,\n                'withdraw': true,\n            },\n            'timeframes': {\n                '1d': '1year',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766049-2b294408-5ecc-11e7-85cc-adaff013dc1a.jpg',\n                'api': 'https://1btcxe.com/api',\n                'www': 'https://1btcxe.com',\n                'doc': 'https://1btcxe.com/api-docs.php',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'stats',\n                        'historical-prices',\n                        'order-book',\n                        'transactions',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balances-and-info',\n                        'open-orders',\n                        'user-transactions',\n                        'btc-deposit-address/get',\n                        'btc-deposit-address/new',\n                        'deposits/get',\n                        'withdrawals/get',\n                        'orders/new',\n                        'orders/edit',\n                        'orders/cancel',\n                        'orders/status',\n                        'withdrawals/new',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/USD': { 'id': 'USD', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD' },\n                'BTC/EUR': { 'id': 'EUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR' },\n                'BTC/CNY': { 'id': 'CNY', 'symbol': 'BTC/CNY', 'base': 'BTC', 'quote': 'CNY' },\n                'BTC/RUB': { 'id': 'RUB', 'symbol': 'BTC/RUB', 'base': 'BTC', 'quote': 'RUB' },\n                'BTC/CHF': { 'id': 'CHF', 'symbol': 'BTC/CHF', 'base': 'BTC', 'quote': 'CHF' },\n                'BTC/JPY': { 'id': 'JPY', 'symbol': 'BTC/JPY', 'base': 'BTC', 'quote': 'JPY' },\n                'BTC/GBP': { 'id': 'GBP', 'symbol': 'BTC/GBP', 'base': 'BTC', 'quote': 'GBP' },\n                'BTC/CAD': { 'id': 'CAD', 'symbol': 'BTC/CAD', 'base': 'BTC', 'quote': 'CAD' },\n                'BTC/AUD': { 'id': 'AUD', 'symbol': 'BTC/AUD', 'base': 'BTC', 'quote': 'AUD' },\n                'BTC/AED': { 'id': 'AED', 'symbol': 'BTC/AED', 'base': 'BTC', 'quote': 'AED' },\n                'BTC/BGN': { 'id': 'BGN', 'symbol': 'BTC/BGN', 'base': 'BTC', 'quote': 'BGN' },\n                'BTC/CZK': { 'id': 'CZK', 'symbol': 'BTC/CZK', 'base': 'BTC', 'quote': 'CZK' },\n                'BTC/DKK': { 'id': 'DKK', 'symbol': 'BTC/DKK', 'base': 'BTC', 'quote': 'DKK' },\n                'BTC/HKD': { 'id': 'HKD', 'symbol': 'BTC/HKD', 'base': 'BTC', 'quote': 'HKD' },\n                'BTC/HRK': { 'id': 'HRK', 'symbol': 'BTC/HRK', 'base': 'BTC', 'quote': 'HRK' },\n                'BTC/HUF': { 'id': 'HUF', 'symbol': 'BTC/HUF', 'base': 'BTC', 'quote': 'HUF' },\n                'BTC/ILS': { 'id': 'ILS', 'symbol': 'BTC/ILS', 'base': 'BTC', 'quote': 'ILS' },\n                'BTC/INR': { 'id': 'INR', 'symbol': 'BTC/INR', 'base': 'BTC', 'quote': 'INR' },\n                'BTC/MUR': { 'id': 'MUR', 'symbol': 'BTC/MUR', 'base': 'BTC', 'quote': 'MUR' },\n                'BTC/MXN': { 'id': 'MXN', 'symbol': 'BTC/MXN', 'base': 'BTC', 'quote': 'MXN' },\n                'BTC/NOK': { 'id': 'NOK', 'symbol': 'BTC/NOK', 'base': 'BTC', 'quote': 'NOK' },\n                'BTC/NZD': { 'id': 'NZD', 'symbol': 'BTC/NZD', 'base': 'BTC', 'quote': 'NZD' },\n                'BTC/PLN': { 'id': 'PLN', 'symbol': 'BTC/PLN', 'base': 'BTC', 'quote': 'PLN' },\n                'BTC/RON': { 'id': 'RON', 'symbol': 'BTC/RON', 'base': 'BTC', 'quote': 'RON' },\n                'BTC/SEK': { 'id': 'SEK', 'symbol': 'BTC/SEK', 'base': 'BTC', 'quote': 'SEK' },\n                'BTC/SGD': { 'id': 'SGD', 'symbol': 'BTC/SGD', 'base': 'BTC', 'quote': 'SGD' },\n                'BTC/THB': { 'id': 'THB', 'symbol': 'BTC/THB', 'base': 'BTC', 'quote': 'THB' },\n                'BTC/TRY': { 'id': 'TRY', 'symbol': 'BTC/TRY', 'base': 'BTC', 'quote': 'TRY' },\n                'BTC/ZAR': { 'id': 'ZAR', 'symbol': 'BTC/ZAR', 'base': 'BTC', 'quote': 'ZAR' },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostBalancesAndInfo ();\n        let balance = response['balances-and-info'];\n        let result = { 'info': balance };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            account['free'] = this.safeFloat (balance['available'], currency, 0.0);\n            account['used'] = this.safeFloat (balance['on_hold'], currency, 0.0);\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let response = await this.publicGetOrderBook (this.extend ({\n            'currency': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (response['order-book'], undefined, 'bid', 'ask', 'price', 'order_amount');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let response = await this.publicGetStats (this.extend ({\n            'currency': this.marketId (symbol),\n        }, params));\n        let ticker = response['stats'];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['max']),\n            'low': parseFloat (ticker['min']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': parseFloat (ticker['open']),\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last_price']),\n            'change': parseFloat (ticker['daily_change']),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': parseFloat (ticker['total_btc_traded']),\n        };\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1d', since = undefined, limit = undefined) {\n        return [\n            this.parse8601 (ohlcv['date'] + ' 00:00:00'),\n            undefined,\n            undefined,\n            undefined,\n            parseFloat (ohlcv['price']),\n            undefined,\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1d', since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetHistoricalPrices (this.extend ({\n            'currency': market['id'],\n            'timeframe': this.timeframes[timeframe],\n        }, params));\n        let ohlcvs = this.omit (response['historical-prices'], 'request_currency');\n        return this.parseOHLCVs (ohlcvs, market, timeframe, since, limit);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['timestamp']) * 1000;\n        return {\n            'id': trade['id'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': undefined,\n            'type': undefined,\n            'side': trade['maker_type'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetTransactions (this.extend ({\n            'currency': market['id'],\n        }, params));\n        let trades = this.omit (response['transactions'], 'request_currency');\n        return this.parseTrades (trades, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let order = {\n            'side': side,\n            'type': type,\n            'currency': this.marketId (symbol),\n            'amount': amount,\n        };\n        if (type === 'limit')\n            order['limit_price'] = price;\n        let result = await this.privatePostOrdersNew (this.extend (order, params));\n        return {\n            'info': result,\n            'id': result,\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostOrdersCancel ({ 'id': id });\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostWithdrawalsNew (this.extend ({\n            'currency': currency,\n            'amount': parseFloat (amount),\n            'address': address,\n        }, params));\n        return {\n            'info': response,\n            'id': response['result']['uuid'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        if (this.id === 'cryptocapital')\n            throw new ExchangeError (this.id + ' is an abstract base API for _1btcxe');\n        let url = this.urls['api'] + '/' + path;\n        if (api === 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let query = this.extend ({\n                'api_key': this.apiKey,\n                'nonce': this.nonce (),\n            }, params);\n            let request = this.json (query);\n            query['signature'] = this.hmac (this.encode (request), this.encode (this.secret));\n            body = this.json (query);\n            headers = { 'Content-Type': 'application/json' };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('errors' in response) {\n            let errors = [];\n            for (let e = 0; e < response['errors'].length; e++) {\n                let error = response['errors'][e];\n                errors.push (error['code'] + ': ' + error['message']);\n            }\n            errors = errors.join (' ');\n            throw new ExchangeError (this.id + ' ' + errors);\n        }\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, OrderNotFound } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class acx extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'acx',\n            'name': 'ACX',\n            'countries': 'AU',\n            'rateLimit': 1000,\n            'version': 'v2',\n            'has': {\n                'CORS': true,\n                'fetchTickers': true,\n                'fetchOHLCV': true,\n                'withdraw': true,\n            },\n            'timeframes': {\n                '1m': '1',\n                '5m': '5',\n                '15m': '15',\n                '30m': '30',\n                '1h': '60',\n                '2h': '120',\n                '4h': '240',\n                '12h': '720',\n                '1d': '1440',\n                '3d': '4320',\n                '1w': '10080',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30247614-1fe61c74-9621-11e7-9e8c-f1a627afa279.jpg',\n                'extension': '.json',\n                'api': 'https://acx.io/api',\n                'www': 'https://acx.io',\n                'doc': 'https://acx.io/documents/api_v2',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'markets', // Get all available markets\n                        'tickers', // Get ticker of all markets\n                        'tickers/{market}', // Get ticker of specific market\n                        'trades', // Get recent trades on market, each trade is included only once Trades are sorted in reverse creation order.\n                        'order_book', // Get the order book of specified market\n                        'depth', // Get depth or specified market Both asks and bids are sorted from highest price to lowest.\n                        'k', // Get OHLC(k line) of specific market\n                        'k_with_pending_trades', // Get K data with pending trades, which are the trades not included in K data yet, because there's delay between trade generated and processed by K data generator\n                        'timestamp', // Get server current time, in seconds since Unix epoch\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'members/me', // Get your profile and accounts info\n                        'deposits', // Get your deposits history\n                        'deposit', // Get details of specific deposit\n                        'deposit_address', // Where to deposit The address field could be empty when a new address is generating (e.g. for bitcoin), you should try again later in that case.\n                        'orders', // Get your orders, results is paginated\n                        'order', // Get information of specified order\n                        'trades/my', // Get your executed trades Trades are sorted in reverse creation order.\n                        'withdraws', // Get your cryptocurrency withdraws\n                        'withdraw', // Get your cryptocurrency withdraw\n                    ],\n                    'post': [\n                        'orders', // Create a Sell/Buy order\n                        'orders/multi', // Create multiple sell/buy orders\n                        'orders/clear', // Cancel all my orders\n                        'order/delete', // Cancel an order\n                        'withdraw', // Create a withdraw\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'maker': 0.0,\n                    'taker': 0.0,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'withdraw': 0.0, // There is only 1% fee on withdrawals to your bank account.\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetMarkets ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['id'];\n            let symbol = market['name'];\n            let [ base, quote ] = symbol.split ('/');\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetMembersMe ();\n        let balances = response['accounts'];\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let uppercase = currency.toUpperCase ();\n            let account = {\n                'free': parseFloat (balance['balance']),\n                'used': parseFloat (balance['locked']),\n                'total': 0.0,\n            };\n            account['total'] = this.sum (account['free'], account['used']);\n            result[uppercase] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orderbook = await this.publicGetDepth (this.extend ({\n            'market': market['id'],\n            'limit': 300,\n        }, params));\n        let timestamp = orderbook['timestamp'] * 1000;\n        let result = this.parseOrderBook (orderbook, timestamp);\n        result['bids'] = this.sortBy (result['bids'], 0, true);\n        result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['at'] * 1000;\n        ticker = ticker['ticker'];\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high', undefined),\n            'low': this.safeFloat (ticker, 'low', undefined),\n            'bid': this.safeFloat (ticker, 'buy', undefined),\n            'ask': this.safeFloat (ticker, 'sell', undefined),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last', undefined),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'vol', undefined),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTickers (params);\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = undefined;\n            let symbol = id;\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            } else {\n                let base = id.slice (0, 3);\n                let quote = id.slice (3, 6);\n                base = base.toUpperCase ();\n                quote = quote.toUpperCase ();\n                base = this.commonCurrencyCode (base);\n                quote = this.commonCurrencyCode (quote);\n                symbol = base + '/' + quote;\n            }\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTickersMarket (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseTicker (response, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = this.parse8601 (trade['created_at']);\n        return {\n            'id': trade['id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': this.safeFloat (trade, 'price'),\n            'amount': this.safeFloat (trade, 'volume'),\n            'cost': this.safeFloat (trade, 'funds'),\n            'info': trade,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTrades (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv[0] * 1000,\n            ohlcv[1],\n            ohlcv[2],\n            ohlcv[3],\n            ohlcv[4],\n            ohlcv[5],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        if (!limit)\n            limit = 500; // default is 30\n        let request = {\n            'market': market['id'],\n            'period': this.timeframes[timeframe],\n            'limit': limit,\n        };\n        if (since)\n            request['timestamp'] = since;\n        let response = await this.publicGetK (this.extend (request, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    parseOrder (order, market = undefined) {\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else {\n            let marketId = order['market'];\n            symbol = this.marketsById[marketId]['symbol'];\n        }\n        let timestamp = this.parse8601 (order['created_at']);\n        let state = order['state'];\n        let status = undefined;\n        if (state === 'done') {\n            status = 'closed';\n        } else if (state === 'wait') {\n            status = 'open';\n        } else if (state === 'cancel') {\n            status = 'canceled';\n        }\n        return {\n            'id': order['id'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': status,\n            'symbol': symbol,\n            'type': order['ord_type'],\n            'side': order['side'],\n            'price': parseFloat (order['price']),\n            'amount': parseFloat (order['volume']),\n            'filled': parseFloat (order['executed_volume']),\n            'remaining': parseFloat (order['remaining_volume']),\n            'trades': undefined,\n            'fee': undefined,\n            'info': order,\n        };\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'market': this.marketId (symbol),\n            'side': side,\n            'volume': amount.toString (),\n            'ord_type': type,\n        };\n        if (type === 'limit') {\n            order['price'] = price.toString ();\n        }\n        let response = await this.privatePostOrders (this.extend (order, params));\n        let market = this.marketsById[response['market']];\n        return this.parseOrder (response, market);\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let result = await this.privatePostOrderDelete ({ 'id': id });\n        let order = this.parseOrder (result);\n        if (order['status'] === 'closed') {\n            throw new OrderNotFound (this.id + ' ' + result);\n        }\n        return order;\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let result = await this.privatePostWithdraw (this.extend ({\n            'currency': currency.toLowerCase (),\n            'sum': amount,\n            'address': address,\n        }, params));\n        return {\n            'info': result,\n            'id': undefined,\n        };\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    encodeParams (params) {\n        if ('orders' in params) {\n            let orders = params['orders'];\n            let query = this.urlencode (this.keysort (this.omit (params, 'orders')));\n            for (let i = 0; i < orders.length; i++) {\n                let order = orders[i];\n                let keys = Object.keys (order);\n                for (let k = 0; k < keys.length; k++) {\n                    let key = keys[k];\n                    let value = order[key];\n                    query += '&orders%5B%5D%5B' + key + '%5D=' + value.toString ();\n                }\n            }\n            return query;\n        }\n        return this.urlencode (this.keysort (params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let request = '/api' + '/' + this.version + '/' + this.implodeParams (path, params);\n        if ('extension' in this.urls)\n            request += this.urls['extension'];\n        let query = this.omit (params, this.extractParams (path));\n        let url = this.urls['api'] + request;\n        if (api === 'public') {\n            if (Object.keys (query).length) {\n                url += '?' + this.urlencode (query);\n            }\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let query = this.encodeParams (this.extend ({\n                'access_key': this.apiKey,\n                'tonce': nonce,\n            }, params));\n            let auth = method + '|' + request + '|' + query;\n            let signature = this.hmac (this.encode (auth), this.encode (this.secret));\n            let suffix = query + '&signature=' + signature;\n            if (method === 'GET') {\n                url += '?' + suffix;\n            } else {\n                body = suffix;\n                headers = { 'Content-Type': 'application/x-www-form-urlencoded' };\n            }\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst okcoinusd = require ('./okcoinusd.js');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class allcoin extends okcoinusd {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'allcoin',\n            'name': 'Allcoin',\n            'countries': 'CA',\n            'has': {\n                'CORS': false,\n            },\n            'extension': '',\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/31561809-c316b37c-b061-11e7-8d5a-b547b4d730eb.jpg',\n                'api': {\n                    'web': 'https://www.allcoin.com',\n                    'public': 'https://api.allcoin.com/api',\n                    'private': 'https://api.allcoin.com/api',\n                },\n                'www': 'https://www.allcoin.com',\n                'doc': 'https://www.allcoin.com/About/APIReference',\n            },\n            'api': {\n                'web': {\n                    'get': [\n                        'Home/MarketOverViewDetail/',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'depth',\n                        'kline',\n                        'ticker',\n                        'trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'batch_trade',\n                        'cancel_order',\n                        'order_history',\n                        'order_info',\n                        'orders_info',\n                        'repayment',\n                        'trade',\n                        'trade_history',\n                        'userinfo',\n                    ],\n                },\n            },\n            'markets': undefined,\n        });\n    }\n\n    async fetchMarkets () {\n        let result = [];\n        let response = await this.webGetHomeMarketOverViewDetail ();\n        let coins = response['marketCoins'];\n        for (let j = 0; j < coins.length; j++) {\n            let markets = coins[j]['Markets'];\n            for (let k = 0; k < markets.length; k++) {\n                let market = markets[k]['Market'];\n                let base = market['Primary'];\n                let quote = market['Secondary'];\n                let id = base.toLowerCase () + '_' + quote.toLowerCase ();\n                let symbol = base + '/' + quote;\n                result.push ({\n                    'id': id,\n                    'symbol': symbol,\n                    'base': base,\n                    'quote': quote,\n                    'type': 'spot',\n                    'spot': true,\n                    'future': false,\n                    'info': market,\n                });\n            }\n        }\n        return result;\n    }\n\n    parseOrderStatus (status) {\n        if (status === -1)\n            return 'canceled';\n        if (status === 0)\n            return 'open';\n        if (status === 1)\n            return 'open'; // partially filled\n        if (status === 2)\n            return 'closed';\n        if (status === 10)\n            return 'canceled';\n        return status;\n    }\n\n    getCreateDateField () {\n        // allcoin typo create_data instead of create_date\n        return 'create_data';\n    }\n\n    getOrdersField () {\n        // allcoin typo order instead of orders (expected based on their API docs)\n        return 'order';\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class anxpro extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'anxpro',\n            'name': 'ANXPro',\n            'countries': [ 'JP', 'SG', 'HK', 'NZ' ],\n            'version': '2',\n            'rateLimit': 1500,\n            'has': {\n                'CORS': false,\n                'fetchTrades': false,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27765983-fd8595da-5ec9-11e7-82e3-adb3ab8c2612.jpg',\n                'api': 'https://anxpro.com/api',\n                'www': 'https://anxpro.com',\n                'doc': [\n                    'http://docs.anxv2.apiary.io',\n                    'https://anxpro.com/pages/api',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        '{currency_pair}/money/ticker',\n                        '{currency_pair}/money/depth/full',\n                        '{currency_pair}/money/trade/fetch', // disabled by ANXPro\n                    ],\n                },\n                'private': {\n                    'post': [\n                        '{currency_pair}/money/order/add',\n                        '{currency_pair}/money/order/cancel',\n                        '{currency_pair}/money/order/quote',\n                        '{currency_pair}/money/order/result',\n                        '{currency_pair}/money/orders',\n                        'money/{currency}/address',\n                        'money/{currency}/send_simple',\n                        'money/info',\n                        'money/trade/list',\n                        'money/wallet/history',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/USD': { 'id': 'BTCUSD', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD', 'multiplier': 100000 },\n                'BTC/HKD': { 'id': 'BTCHKD', 'symbol': 'BTC/HKD', 'base': 'BTC', 'quote': 'HKD', 'multiplier': 100000 },\n                'BTC/EUR': { 'id': 'BTCEUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR', 'multiplier': 100000 },\n                'BTC/CAD': { 'id': 'BTCCAD', 'symbol': 'BTC/CAD', 'base': 'BTC', 'quote': 'CAD', 'multiplier': 100000 },\n                'BTC/AUD': { 'id': 'BTCAUD', 'symbol': 'BTC/AUD', 'base': 'BTC', 'quote': 'AUD', 'multiplier': 100000 },\n                'BTC/SGD': { 'id': 'BTCSGD', 'symbol': 'BTC/SGD', 'base': 'BTC', 'quote': 'SGD', 'multiplier': 100000 },\n                'BTC/JPY': { 'id': 'BTCJPY', 'symbol': 'BTC/JPY', 'base': 'BTC', 'quote': 'JPY', 'multiplier': 100000 },\n                'BTC/GBP': { 'id': 'BTCGBP', 'symbol': 'BTC/GBP', 'base': 'BTC', 'quote': 'GBP', 'multiplier': 100000 },\n                'BTC/NZD': { 'id': 'BTCNZD', 'symbol': 'BTC/NZD', 'base': 'BTC', 'quote': 'NZD', 'multiplier': 100000 },\n                'LTC/BTC': { 'id': 'LTCBTC', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC', 'multiplier': 100000 },\n                'STR/BTC': { 'id': 'STRBTC', 'symbol': 'STR/BTC', 'base': 'STR', 'quote': 'BTC', 'multiplier': 100000000 },\n                'XRP/BTC': { 'id': 'XRPBTC', 'symbol': 'XRP/BTC', 'base': 'XRP', 'quote': 'BTC', 'multiplier': 100000000 },\n                'DOGE/BTC': { 'id': 'DOGEBTC', 'symbol': 'DOGE/BTC', 'base': 'DOGE', 'quote': 'BTC', 'multiplier': 100000000 },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.3 / 100,\n                    'taker': 0.6 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostMoneyInfo ();\n        let balance = response['data'];\n        let currencies = Object.keys (balance['Wallets']);\n        let result = { 'info': balance };\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let account = this.account ();\n            if (currency in balance['Wallets']) {\n                let wallet = balance['Wallets'][currency];\n                account['free'] = parseFloat (wallet['Available_Balance']['value']);\n                account['total'] = parseFloat (wallet['Balance']['value']);\n                account['used'] = account['total'] - account['free'];\n            }\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let response = await this.publicGetCurrencyPairMoneyDepthFull (this.extend ({\n            'currency_pair': this.marketId (symbol),\n        }, params));\n        let orderbook = response['data'];\n        let t = parseInt (orderbook['dataUpdateTime']);\n        let timestamp = parseInt (t / 1000);\n        return this.parseOrderBook (orderbook, timestamp, 'bids', 'asks', 'price', 'amount');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let response = await this.publicGetCurrencyPairMoneyTicker (this.extend ({\n            'currency_pair': this.marketId (symbol),\n        }, params));\n        let ticker = response['data'];\n        let t = parseInt (ticker['dataUpdateTime']);\n        let timestamp = parseInt (t / 1000);\n        let bid = this.safeFloat (ticker['buy'], 'value');\n        let ask = this.safeFloat (ticker['sell'], 'value');\n        let baseVolume = parseFloat (ticker['vol']['value']);\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']['value']),\n            'low': parseFloat (ticker['low']['value']),\n            'bid': bid,\n            'ask': ask,\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']['value']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': parseFloat (ticker['avg']['value']),\n            'baseVolume': baseVolume,\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        throw new ExchangeError (this.id + ' switched off the trades endpoint, see their docs at http://docs.anxv2.apiary.io/reference/market-data/currencypairmoneytradefetch-disabled');\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let market = this.market (symbol);\n        let order = {\n            'currency_pair': market['id'],\n            'amount_int': parseInt (amount * 100000000), // 10^8\n        };\n        if (type === 'limit') {\n            order['price_int'] = parseInt (price * market['multiplier']); // 10^5 or 10^8\n        }\n        order['type'] = (side === 'buy') ? 'bid' : 'ask';\n        let result = await this.privatePostCurrencyPairMoneyOrderAdd (this.extend (order, params));\n        return {\n            'info': result,\n            'id': result['data'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCurrencyPairMoneyOrderCancel ({ 'oid': id });\n    }\n\n    getAmountMultiplier (currency) {\n        if (currency === 'BTC') {\n            return 100000000;\n        } else if (currency === 'LTC') {\n            return 100000000;\n        } else if (currency === 'STR') {\n            return 100000000;\n        } else if (currency === 'XRP') {\n            return 100000000;\n        } else if (currency === 'DOGE') {\n            return 100000000;\n        }\n        return 100;\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let multiplier = this.getAmountMultiplier (currency);\n        let response = await this.privatePostMoneyCurrencySendSimple (this.extend ({\n            'currency': currency,\n            'amount_int': parseInt (amount * multiplier),\n            'address': address,\n        }, params));\n        return {\n            'info': response,\n            'id': response['data']['transactionId'],\n        };\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let request = this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        let url = this.urls['api'] + '/' + this.version + '/' + request;\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({ 'nonce': nonce }, query));\n            let secret = this.base64ToBinary (this.secret);\n            let auth = request + '\\0' + body;\n            let signature = this.hmac (this.encode (auth), secret, 'sha512', 'base64');\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Rest-Key': this.apiKey,\n                'Rest-Sign': this.decode (signature),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('result' in response)\n            if (response['result'] === 'success')\n                return response;\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","\"use strict\";\n\n/*  ------------------------------------------------------------------------ */\n\nconst functions = require ('./functions')\n    , Market    = require ('./Market')\n\nconst { isNode\n      , keys\n      , values\n      , deepExtend\n      , extend\n      , flatten\n      , indexBy\n      , sortBy\n      , groupBy\n      , aggregate\n      , uuid\n      , unCamelCase\n      , precisionFromString\n      , throttle\n      , capitalize } = functions\n\nconst { now\n      , sleep\n      , timeout\n      , TimedOut } = require ('./functions/time')\n\nconst { ExchangeError\n      , NotSupported\n      , AuthenticationError\n      , DDoSProtection\n      , RequestTimeout\n      , ExchangeNotAvailable } = require ('./errors')\n\nconst defaultFetch = isNode ? require ('fetch-ponyfill')().fetch : fetch\n\nconst journal = undefined // isNode && require ('./journal') // stub until we get a better solution for Webpack and React\n\n/*  ------------------------------------------------------------------------ */\n\nmodule.exports = class Exchange {\n\n    getMarket (symbol) {\n\n        if (!this.marketClasses)\n            this.marketClasses = {}\n\n        let marketClass = this.marketClasses[symbol]\n\n        if (marketClass)\n            return marketClass\n\n        marketClass = new Market (this, symbol)\n        this.marketClasses[symbol] = marketClass // only one Market instance per market\n        return marketClass\n    }\n\n    describe () { return {} }\n\n    constructor (userConfig = {}) {\n\n        Object.assign (this, functions, { encode: string => string, decode: string => string })\n\n        if (isNode)\n            this.nodeVersion = process.version.match (/\\d+\\.\\d+.\\d+/) [0]\n\n        // this.initRestRateLimiter ()\n\n        // if (isNode) {\n        //     this.userAgent = {\n        //         'User-Agent': 'ccxt/' + Exchange.ccxtVersion +\n        //             ' (+https://github.com/ccxt/ccxt)' +\n        //             ' Node.js/' + this.nodeVersion + ' (JavaScript)'\n        //     }\n        // }\n\n        this.userAgents = {\n            'chrome': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/62.0.3202.94 Safari/537.36',\n            'chrome39': 'Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/39.0.2171.71 Safari/537.36',\n        }\n\n        this.headers = {}\n\n        // prepended to URL, like https://proxy.com/https://exchange.com/api...\n        this.proxy = ''\n        this.origin = '*' // CORS origin\n\n        this.iso8601          = timestamp => new Date (timestamp).toISOString ()\n        this.parse8601        = x => Date.parse (((x.indexOf ('+') >= 0) || (x.slice (-1) == 'Z')) ? x : (x + 'Z'))\n        this.milliseconds     = now\n        this.microseconds     = () => now () * 1000 // TODO: utilize performance.now for that purpose\n        this.seconds          = () => Math.floor (now () / 1000)\n        this.id               = undefined\n\n        // rate limiter settings\n        this.enableRateLimit  = false\n        this.rateLimit        = 2000  // milliseconds = seconds * 1000\n\n        this.parseJsonResponse             = true  // whether a reply is required to be in JSON or not\n        this.substituteCommonCurrencyCodes = true  // reserved\n        this.parseBalanceFromOpenOrders    = false // some exchanges return balance updates from order API endpoints\n\n        // do not delete this line, it is needed for users to be able to define their own fetchImplementation\n        this.fetchImplementation = defaultFetch\n\n        this.timeout          = 10000 // milliseconds\n        this.verbose          = false\n        this.debug            = false\n        this.journal          = 'debug.json'\n        this.userAgent        = undefined\n        this.twofa            = false // two-factor authentication (2FA)\n        this.timeframes       = undefined\n\n        this.apiKey   = undefined\n        this.secret   = undefined\n        this.uid      = undefined\n        this.login    = undefined\n        this.password = undefined\n\n        this.requiredCredentials = {\n            'apiKey':   true,\n            'secret':   true,\n            'uid':      false,\n            'login':    false,\n            'password': false,\n        }\n\n        this.exceptions = {}\n        this.balance    = {}\n        this.orderbooks = {}\n        this.tickers    = {}\n        this.fees       = {}\n        this.orders     = {}\n        this.trades     = {}\n        this.currencies = {}\n\n        this.last_http_response = undefined\n        this.last_json_response = undefined\n\n        this.arrayConcat = (a, b) => a.concat (b)\n\n        const names = Object.getOwnPropertyNames (this).concat (\n                      Object.getOwnPropertyNames (this.constructor.prototype))\n\n        for (const k of names)\n            this[unCamelCase (k)] = this[k]\n\n    /*  exchange's capabilities (overrideable)      */\n\n        this.has = {\n            'CORS': false,\n            'publicAPI': true,\n            'privateAPI': true,\n            'cancelOrder': true,\n            'createDepositAddress': false,\n            'createOrder': true,\n            'deposit': false,\n            'fetchBalance': true,\n            'fetchClosedOrders': false,\n            'fetchCurrencies': false,\n            'fetchDepositAddress': false,\n            'fetchMarkets': true,\n            'fetchMyTrades': false,\n            'fetchOHLCV': false,\n            'fetchOpenOrders': false,\n            'fetchOrder': false,\n            'fetchOrderBook': true,\n            'fetchOrders': false,\n            'fetchTicker': true,\n            'fetchTickers': false,\n            'fetchBidsAsks': false,\n            'fetchTrades': true,\n            'withdraw': false,\n        }\n\n        // merge configs\n        const config = deepExtend (this.describe (), userConfig)\n\n        // merge to this\n        for (const [property, value] of Object.entries (config))\n            this[property] = deepExtend (this[property], value)\n\n        // generate old metainfo interface\n        for (const k in this.has) {\n            this['has' + capitalize (k)] = !!this.has[k] // converts 'emulated' to true\n        }\n\n        if (this.api)\n            this.defineRestApi (this.api, 'request')\n\n        this.initRestRateLimiter ()\n\n        if (this.markets)\n            this.setMarkets (this.markets)\n\n        if (this.debug && journal) {\n            journal (() => this.journal, this, Object.keys (this.has))\n        }\n    }\n\n    defaults () {\n        return { /* override me */ }\n    }\n\n    nonce () {\n        return this.seconds ()\n    }\n\n    encodeURIComponent (...args) {\n        return encodeURIComponent (...args)\n    }\n\n    checkRequiredCredentials () {\n        Object.keys (this.requiredCredentials).map (key => {\n            if (this.requiredCredentials[key] && !this[key])\n                throw new AuthenticationError (this.id + ' requires `' + key + '`')\n        })\n    }\n\n    initRestRateLimiter () {\n\n        const fetchImplementation = this.fetchImplementation\n\n        this.tokenBucket = this.extend ({\n            refillRate:  1 / this.rateLimit,\n            delay:       1,\n            capacity:    1,\n            defaultCost: 1,\n            maxCapacity: 1000,\n        }, this.tokenBucket)\n\n        this.throttle = throttle (this.tokenBucket)\n\n        this.executeRestRequest = function (url, method = 'GET', headers = undefined, body = undefined) {\n\n            let promise =\n                fetchImplementation (url, { 'method': method, 'headers': headers, 'body': body, 'agent': this.tunnelAgent || null, timeout: this.timeout})\n                    .catch (e => {\n                        if (isNode)\n                            throw new ExchangeNotAvailable ([ this.id, method, url, e.type, e.message ].join (' '))\n                        throw e // rethrow all unknown errors\n                    })\n                    .then (response => this.handleRestErrors (response, url, method, headers, body))\n                    .then (response => this.handleRestResponse (response, url, method, headers, body))\n\n            return timeout (this.timeout, promise).catch (e => {\n                if (e instanceof TimedOut)\n                    throw new RequestTimeout (this.id + ' ' + method + ' ' + url + ' request timed out (' + this.timeout + ' ms)')\n                throw e\n            })\n        }\n    }\n\n    defineRestApi (api, methodName, options = {}) {\n\n        for (const type of Object.keys (api)) {\n            for (const httpMethod of Object.keys (api[type])) {\n\n                let urls = api[type][httpMethod]\n                for (let i = 0; i < urls.length; i++) {\n                    let url = urls[i].trim ()\n                    let splitPath = url.split (/[^a-zA-Z0-9]/)\n\n                    let uppercaseMethod  = httpMethod.toUpperCase ()\n                    let lowercaseMethod  = httpMethod.toLowerCase ()\n                    let camelcaseMethod  = this.capitalize (lowercaseMethod)\n                    let camelcaseSuffix  = splitPath.map (this.capitalize).join ('')\n                    let underscoreSuffix = splitPath.map (x => x.trim ().toLowerCase ()).filter (x => x.length > 0).join ('_')\n\n                    if (camelcaseSuffix.indexOf (camelcaseMethod) === 0)\n                        camelcaseSuffix = camelcaseSuffix.slice (camelcaseMethod.length)\n\n                    if (underscoreSuffix.indexOf (lowercaseMethod) === 0)\n                        underscoreSuffix = underscoreSuffix.slice (lowercaseMethod.length)\n\n                    let camelcase  = type + camelcaseMethod + this.capitalize (camelcaseSuffix)\n                    let underscore = type + '_' + lowercaseMethod + '_' + underscoreSuffix\n\n                    if ('suffixes' in options) {\n                        if ('camelcase' in options['suffixes'])\n                            camelcase += options['suffixes']['camelcase']\n                        if ('underscore' in options.suffixes)\n                            underscore += options['suffixes']['underscore']\n                    }\n\n                    if ('underscore_suffix' in options)\n                        underscore += options.underscoreSuffix;\n                    if ('camelcase_suffix' in options)\n                        camelcase += options.camelcaseSuffix;\n\n                    let partial = async params => this[methodName] (url, type, uppercaseMethod, params || {})\n\n                    this[camelcase]  = partial\n                    this[underscore] = partial\n                }\n            }\n        }\n    }\n\n    fetch (url, method = 'GET', headers = undefined, body = undefined) {\n\n        if (isNode && this.userAgent) {\n            if (typeof this.userAgent == 'string')\n                headers = extend ({ 'User-Agent': this.userAgent }, headers)\n            else if ((typeof this.userAgent == 'object') && ('User-Agent' in this.userAgent))\n                headers = extend (this.userAgent, headers)\n        }\n\n        if (typeof this.proxy == 'function') {\n\n            url = this.proxy (url)\n            if (isNode)\n                headers = extend ({ 'Origin': this.origin }, headers)\n\n        } else if (typeof this.proxy == 'string') {\n\n            if (this.proxy.length)\n                if (isNode)\n                    headers = extend ({ 'Origin': this.origin }, headers)\n\n            url = this.proxy + url\n        }\n\n        headers = extend (this.headers, headers)\n\n        if (this.verbose)\n            console.log (this.id, method, url, \"\\nRequest:\\n\", headers, body)\n\n        return this.executeRestRequest (url, method, headers, body)\n    }\n\n    async fetch2 (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n\n        if (this.enableRateLimit)\n            await this.throttle ()\n\n        let request = this.sign (path, api, method, params, headers, body)\n        return this.fetch (request.url, request.method, request.headers, request.body)\n    }\n\n    request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        return this.fetch2 (path, api, method, params, headers, body)\n    }\n\n    handleErrors (statusCode, statusText, url, method, headers, body) {\n        // override me\n    }\n\n    defaultErrorHandler (code, reason, url, method, headers, body) {\n        if ((code >= 200) && (code <= 300))\n            return body\n        let error = undefined\n        this.last_http_response = body\n        let details = body\n        let match = body.match (/\\<title\\>([^<]+)/i)\n        if (match)\n            details = match[1].trim ();\n        if ([ 418, 429 ].includes (code)) {\n            error = DDoSProtection\n        } else if ([ 404, 409, 500, 501, 502, 520, 521, 522, 525 ].includes (code)) {\n            error = ExchangeNotAvailable\n        } else if ([ 400, 403, 405, 503, 530 ].includes (code)) {\n            let ddosProtection = body.match (/cloudflare|incapsula/i)\n            if (ddosProtection) {\n                error = DDoSProtection\n            } else {\n                error = ExchangeNotAvailable\n                details += ' (possible reasons: ' + [\n                    'invalid API keys',\n                    'bad or old nonce',\n                    'exchange is down or offline',\n                    'on maintenance',\n                    'DDoS protection',\n                    'rate-limiting',\n                ].join (', ') + ')'\n            }\n        } else if ([ 408, 504 ].includes (code)) {\n            error = RequestTimeout\n        } else if ([ 401, 511 ].includes (code)) {\n            error = AuthenticationError\n        } else {\n            error = ExchangeError\n        }\n        throw new error ([ this.id, method, url, code, reason, details ].join (' '))\n    }\n\n    handleRestErrors (response, url, method = 'GET', headers = undefined, body = undefined) {\n\n        if (typeof response === 'string')\n            return response\n\n        return response.text ().then (text => {\n\n            const args = [ response.status, response.statusText, url, method, headers, text ]\n\n            if (this.verbose)\n                console.log (this.id, method, url, response.status, response.statusText, headers, text ? (\"\\nResponse:\\n\" + text) : '')\n\n            this.handleErrors (...args)\n            return this.defaultErrorHandler (...args)\n        })\n    }\n\n    handleRestResponse (response, url, method = 'GET', headers = undefined, body = undefined) {\n\n        try {\n\n            this.last_http_response = response\n            if (this.parseJsonResponse) {\n                this.last_json_response =\n                    ((typeof response === 'string') && (response.length > 1)) ?\n                        JSON.parse (response) : response\n                return this.last_json_response\n            }\n\n            return response\n\n        } catch (e) {\n\n            let maintenance = response.match (/offline|busy|retry|wait|unavailable|maintain|maintenance|maintenancing/i)\n            let ddosProtection = response.match (/cloudflare|incapsula|overload/i)\n\n            if (e instanceof SyntaxError) {\n\n                let error = ExchangeNotAvailable\n                let details = 'not accessible from this location at the moment'\n                if (maintenance)\n                    details = 'offline, on maintenance or unreachable from this location at the moment'\n                if (ddosProtection)\n                    error = DDoSProtection\n                throw new error ([ this.id, method, url, details ].join (' '))\n            }\n\n            if (this.verbose)\n                console.log (this.id, method, url, 'error', e, \"response body:\\n'\" + response + \"'\")\n\n            throw e\n        }\n    }\n\n    setMarkets (markets, currencies = undefined) {\n        let values = Object.values (markets).map (market => deepExtend ({\n            'limits': this.limits,\n            'precision': this.precision,\n        }, this.fees['trading'], market))\n        this.markets = deepExtend (this.markets, indexBy (values, 'symbol'))\n        this.marketsById = indexBy (markets, 'id')\n        this.markets_by_id = this.marketsById\n        this.symbols = Object.keys (this.markets).sort ()\n        this.ids = Object.keys (this.markets_by_id).sort ()\n        if (currencies) {\n            this.currencies = deepExtend (currencies, this.currencies)\n        } else {\n            const baseCurrencies =\n                values.filter (market => 'base' in market)\n                    .map (market => ({\n                        id: market.baseId || market.base,\n                        code: market.base,\n                        precision: market.precision ? (market.precision.base || market.precision.amount) : 8,\n                    }))\n            const quoteCurrencies =\n                values.filter (market => 'quote' in market)\n                    .map (market => ({\n                        id: market.quoteId || market.quote,\n                        code: market.quote,\n                        precision: market.precision ? (market.precision.quote || market.precision.price) : 8,\n                    }))\n            const allCurrencies = baseCurrencies.concat (quoteCurrencies)\n            const groupedCurrencies = groupBy (allCurrencies, 'code')\n            const currencies = Object.keys (groupedCurrencies).map (code =>\n                groupedCurrencies[code].reduce ((previous, current) =>\n                    ((previous.precision > current.precision) ? previous : current), groupedCurrencies[code][0]))\n            const sortedCurrencies = sortBy (flatten (currencies), 'code')\n            this.currencies = deepExtend (indexBy (sortedCurrencies, 'code'), this.currencies)\n        }\n        this.currencies_by_id = indexBy (this.currencies, 'id')\n        return this.markets\n    }\n\n    async loadMarkets (reload = false) {\n        if (!reload && this.markets) {\n            if (!this.marketsById) {\n                return this.setMarkets (this.markets)\n            }\n            return this.markets\n        }\n        const markets = await this.fetchMarkets ()\n        let currencies = undefined\n        if (this.has.fetchCurrencies) {\n            currencies = await this.fetchCurrencies ()\n        }\n        return this.setMarkets (markets, currencies)\n    }\n\n    fetchBidsAsks (symbols = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchBidsAsks not supported yet')\n    }\n\n    fetchTickers (symbols = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchTickers not supported yet')\n    }\n\n    fetchOrder (id, symbol = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchOrder not supported yet');\n    }\n\n    fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchOrders not supported yet');\n    }\n\n    fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchOpenOrders not supported yet');\n    }\n\n    fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchClosedOrders not supported yet');\n    }\n\n    fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchMyTrades not supported yet');\n    }\n\n    fetchCurrencies () {\n        throw new NotSupported (this.id + ' fetchCurrencies not supported yet');\n    }\n\n    fetchMarkets () {\n        return new Promise ((resolve, reject) => resolve (this.markets))\n    }\n\n    async fetchOrderStatus (id, market = undefined) {\n        let order = await this.fetchOrder (id)\n        return order['status']\n    }\n\n    account () {\n        return {\n            'free': 0.0,\n            'used': 0.0,\n            'total': 0.0,\n        }\n    }\n\n    commonCurrencyCode (currency) {\n        if (!this.substituteCommonCurrencyCodes)\n            return currency\n        if (currency == 'XBT')\n            return 'BTC'\n        if (currency == 'BCC')\n            return 'BCH'\n        if (currency == 'DRK')\n            return 'DASH'\n        return currency\n    }\n\n    currency (code) {\n\n        if (typeof this.currencies == 'undefined')\n            return new ExchangeError (this.id + ' currencies not loaded')\n\n        if ((typeof code === 'string') && (code in this.currencies))\n            return this.currencies[code]\n\n        throw new ExchangeError (this.id + ' does not have currency code ' + code)\n    }\n\n\n    market (symbol) {\n\n        if (typeof this.markets == 'undefined')\n            return new ExchangeError (this.id + ' markets not loaded')\n\n        if ((typeof symbol === 'string') && (symbol in this.markets))\n            return this.markets[symbol]\n\n        throw new ExchangeError (this.id + ' does not have market symbol ' + symbol)\n    }\n\n    marketId (symbol) {\n        return this.market (symbol).id || symbol\n    }\n\n    marketIds (symbols) {\n        return symbols.map (symbol => this.marketId(symbol));\n    }\n\n    symbol (symbol) {\n        return this.market (symbol).symbol || symbol\n    }\n\n    extractParams (string) {\n        let re = /{([a-zA-Z0-9_]+?)}/g\n        let matches = []\n        let match\n        while (match = re.exec (string))\n            matches.push (match[1])\n        return matches\n    }\n\n    implodeParams (string, params) {\n        for (let property in params)\n            string = string.replace ('{' + property + '}', params[property])\n        return string\n    }\n\n    url (path, params = {}) {\n        let result = this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path))\n        if (Object.keys (query).length)\n            result += '?' + this.urlencode (query)\n        return result\n    }\n\n    parseBidAsk (bidask, priceKey = 0, amountKey = 1) {\n        let price = parseFloat (bidask[priceKey])\n        let amount = parseFloat (bidask[amountKey])\n        return [ price, amount ]\n    }\n\n    parseBidsAsks (bidasks, priceKey = 0, amountKey = 1) {\n        return Object.values (bidasks || []).map (bidask => this.parseBidAsk (bidask, priceKey, amountKey))\n    }\n\n    async fetchL2OrderBook (symbol, params = {}) {\n        let orderbook = await this.fetchOrderBook (symbol, params)\n        return extend (orderbook, {\n            'bids': aggregate (orderbook.bids),\n            'asks': aggregate (orderbook.asks),\n        })\n    }\n\n    parseOrderBook (orderbook, timestamp = undefined, bidsKey = 'bids', asksKey = 'asks', priceKey = 0, amountKey = 1) {\n        timestamp = timestamp || this.milliseconds ();\n        return {\n            'bids': sortBy ((bidsKey in orderbook) ? this.parseBidsAsks (orderbook[bidsKey], priceKey, amountKey) : [], 0, true),\n            'asks': sortBy ((asksKey in orderbook) ? this.parseBidsAsks (orderbook[asksKey], priceKey, amountKey) : [], 0),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n        }\n    }\n\n    getCurrencyUsedOnOpenOrders (currency) {\n        return Object.values (this.orders).filter (order => (order['status'] == 'open')).reduce ((total, order) => {\n            let symbol = order['symbol'];\n            let market = this.markets[symbol];\n            let amount = order['remaining']\n            if (currency == market['base'] && order['side'] == 'sell') {\n                return total + amount\n            } else if (currency == market['quote'] && order['side'] == 'buy') {\n                return total + (order['cost'] || (order['price'] * amount))\n            } else {\n                return total\n            }\n        }, 0)\n    }\n\n    parseBalance (balance) {\n\n        const currencies = Object.keys (this.omit (balance, 'info'));\n\n        currencies.forEach (currency => {\n\n            if (typeof balance[currency].used == 'undefined') {\n\n                if (this.parseBalanceFromOpenOrders && ('open_orders' in balance['info'])) {\n                    const exchangeOrdersCount = balance['info']['open_orders'];\n                    const cachedOrdersCount = Object.values (this.orders).filter (order => (order['status'] == 'open')).length;\n                    if (cachedOrdersCount == exchangeOrdersCount) {\n                        balance[currency].used = this.getCurrencyUsedOnOpenOrders (currency)\n                        balance[currency].total = balance[currency].used + balance[currency].free\n                    }\n                } else {\n                    balance[currency].used = this.getCurrencyUsedOnOpenOrders (currency)\n                    balance[currency].total = balance[currency].used + balance[currency].free\n                }\n            }\n\n            [ 'free', 'used', 'total' ].forEach (account => {\n                balance[account] = balance[account] || {}\n                balance[account][currency] = balance[currency][account]\n            })\n        })\n\n        return balance\n    }\n\n    async fetchPartialBalance (part, params = {}) {\n        let balance = await this.fetchBalance (params)\n        return balance[part]\n    }\n\n    fetchFreeBalance (params = {}) {\n        return this.fetchPartialBalance ('free', params)\n    }\n\n    fetchUsedBalance (params = {}) {\n        return this.fetchPartialBalance ('used', params)\n    }\n\n    fetchTotalBalance (params = {}) {\n        return this.fetchPartialBalance ('total', params)\n    }\n\n    filterBySinceLimit (array, since = undefined, limit = undefined) {\n        if (since)\n            array = array.filter (entry => entry.timestamp > since)\n        if (limit)\n            array = array.slice (0, limit)\n        return array\n    }\n\n    parseTrades (trades, market = undefined, since = undefined, limit = undefined) {\n        let result = Object.values (trades).map (trade => this.parseTrade (trade, market))\n        result = sortBy (result, 'timestamp', true)\n        return this.filterBySinceLimit (result, since, limit)\n    }\n\n    parseOrders (orders, market = undefined, since = undefined, limit = undefined) {\n        let result = Object.values (orders).map (order => this.parseOrder (order, market))\n        return this.filterBySinceLimit (result, since, limit)\n    }\n\n    filterOrdersBySymbol (orders, symbol = undefined) {\n        let grouped = this.groupBy (orders, 'symbol')\n        if (symbol) {\n            if (symbol in grouped)\n                return grouped[symbol]\n            return []\n        }\n        return orders\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return ohlcv\n    }\n\n    parseOHLCVs (ohlcvs, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        ohlcvs = Object.values (ohlcvs)\n        let result = []\n        for (let i = 0; i < ohlcvs.length; i++) {\n            if (limit && (result.length >= limit))\n                break;\n            let ohlcv = this.parseOHLCV (ohlcvs[i], market, timeframe, since, limit)\n            if (since && (ohlcv[0] < since))\n                continue\n            result.push (ohlcv)\n        }\n        return result\n    }\n\n    editLimitBuyOrder (id, symbol, ...args) {\n        return this.editLimitOrder (id, symbol, 'buy', ...args)\n    }\n\n    editLimitSellOrder (id, symbol, ...args) {\n        return this.editLimitOrder (id, symbol, 'sell', ...args)\n    }\n\n    editLimitOrder (id, symbol, ...args) {\n        return this.editOrder (id, symbol, 'limit', ...args)\n    }\n\n    async editOrder (id, symbol, ...args) {\n        if (!this.enableRateLimit)\n            throw new ExchangeError (this.id + ' editOrder() requires enableRateLimit = true')\n        await this.cancelOrder (id, symbol);\n        return this.createOrder (symbol, ...args)\n    }\n\n    createLimitBuyOrder (symbol, ...args) {\n        return this.createOrder  (symbol, 'limit', 'buy', ...args)\n    }\n\n    createLimitSellOrder (symbol, ...args) {\n        return this.createOrder (symbol, 'limit', 'sell', ...args)\n    }\n\n    createMarketBuyOrder (symbol, amount, params = {}) {\n        return this.createOrder (symbol, 'market', 'buy', amount, undefined, params)\n    }\n\n    createMarketSellOrder (symbol, amount, params = {}) {\n        return this.createOrder (symbol, 'market', 'sell', amount, undefined, params)\n    }\n\n    costToPrecision (symbol, cost) {\n        return parseFloat (cost).toFixed (this.markets[symbol].precision.price)\n    }\n\n    priceToPrecision (symbol, price) {\n        return parseFloat (price).toFixed (this.markets[symbol].precision.price)\n    }\n\n    amountToPrecision (symbol, amount) {\n        return this.truncate (amount, this.markets[symbol].precision.amount)\n    }\n\n    amountToString (symbol, amount) {\n        return this.truncate_to_string (amount, this.markets[symbol].precision.amount)\n    }\n\n    amountToLots (symbol, amount) {\n        return this.amountToPrecision (symbol, Math.floor (amount / this.markets[symbol].lot) * this.markets[symbol].lot)\n    }\n\n    feeToPrecision (symbol, fee) {\n        return parseFloat (fee).toFixed (this.markets[symbol].precision.price)\n    }\n\n    calculateFee (symbol, type, side, amount, price, takerOrMaker = 'taker', params = {}) {\n        let market = this.markets[symbol]\n        let rate = market[takerOrMaker]\n        let cost = parseFloat (this.costToPrecision (symbol, amount * price))\n        return {\n            'type': takerOrMaker,\n            'currency': market['quote'],\n            'rate': rate,\n            'cost': parseFloat (this.feeToPrecision (symbol, rate * cost)),\n        }\n    }\n\n    Ymd (timestamp, infix = ' ') {\n        let date = new Date (timestamp)\n        let Y = date.getUTCFullYear ()\n        let m = date.getUTCMonth () + 1\n        let d = date.getUTCDate ()\n        m = m < 10 ? ('0' + m) : m\n        d = d < 10 ? ('0' + d) : d\n        return Y + '-' + m + '-' + d\n    }\n\n    YmdHMS (timestamp, infix = ' ') {\n        let date = new Date (timestamp)\n        let Y = date.getUTCFullYear ()\n        let m = date.getUTCMonth () + 1\n        let d = date.getUTCDate ()\n        let H = date.getUTCHours ()\n        let M = date.getUTCMinutes ()\n        let S = date.getUTCSeconds ()\n        m = m < 10 ? ('0' + m) : m\n        d = d < 10 ? ('0' + d) : d\n        H = H < 10 ? ('0' + H) : H\n        M = M < 10 ? ('0' + M) : M\n        S = S < 10 ? ('0' + S) : S\n        return Y + '-' + m + '-' + d + infix + H + ':' + M + ':' + S\n    }\n}\n","\"use strict\";\n\nmodule.exports = class Market {\n\n    constructor (exchange, symbol) {\n        this.exchange = exchange;\n        this.symbol = symbol;\n        this.market = exchange.markets[symbol];\n    }\n\n    amountToPrecision (amount) {\n        return this.exchange.amountToPrecision (this.symbol, amount)\n    }\n\n    createLimitBuyOrder(amount, price) {\n        return this.exchange.createLimitBuyOrder (this.symbol, amount, price)\n    }\n\n    createLimitSellOrder(amount, price) {\n        return this.exchange.createLimitSellOrder (this.symbol, amount, price)\n    }\n}\n","/*  ------------------------------------------------------------------------ */\n\nmodule.exports = subclass (\n\n/*  Root class                  */\n\n    Error,\n\n/*  Derived class hierarchy     */\n\n    { 'BaseError':\n        { 'ExchangeError':\n            { 'NotSupported':               {}\n            , 'AuthenticationError':        {}\n            , 'InvalidNonce':               {}\n            , 'InsufficientFunds':          {}\n            , 'InvalidOrder':\n                { 'OrderNotFound':          {}\n                , 'OrderNotCached':         {}\n                , 'CancelPending':          {}\n                }\n            , 'NetworkError':\n                { 'DDoSProtection':         {}\n                , 'RequestTimeout':         {}\n                , 'ExchangeNotAvailable':   {}\n                }\n            } \n        }\n    }\n)\n\n/*  ------------------------------------------------------------------------ */\n\nfunction subclass (BaseClass, classes, namespace = {}) {\n\n    for (const [$class, subclasses] of Object.entries (classes)) {\n\n        const Class = Object.assign (namespace, {\n            \n        /*  By creating a named property, we trick compiler to assign our class constructor function a name.\n            Otherwise, all our error constructors would be shown as [Function: Error] in the debugger! And\n            the super-useful `e.constructor.name` magic wouldn't work — we then would have no chance to\n            obtain a error type string from an error instance programmatically!                               */\n\n            [$class]: class extends BaseClass {\n\n                constructor (message) {\n            \n                    super (message)\n            \n                /*  A workaround to make `instanceof` work on custom Error classes in transpiled ES5.\n                    See my blog post for the explanation of this hack:\n            \n                    https://medium.com/@xpl/javascript-deriving-from-error-properly-8d2f8f315801        */\n            \n                    this.constructor = Class\n                    this.__proto__   = Class.prototype\n                    this.message     = message\n                }\n            }\n\n        })[$class]\n\n        subclass (Class, subclasses, namespace)\n    }\n\n    return namespace\n}\n\n/*  ------------------------------------------------------------------------ */\n","\"use strict\";\n\n/*  ------------------------------------------------------------------------ */\n\nconst { unCamelCase } = require ('./functions/string')\n\nconst unCamelCasePropertyNames = x => {\n                                    for (const k in x) x[unCamelCase (k)] = x[k] // camel_case_method = camelCaseMethod\n                                    return x\n                                }\n\n/*  ------------------------------------------------------------------------ */\n\nmodule.exports = unCamelCasePropertyNames (Object.assign ({}\n\n    , require ('./functions/platform')\n    , require ('./functions/generic')\n    , require ('./functions/string')\n    , require ('./functions/type')\n    , require ('./functions/number')\n    , require ('./functions/encode')\n    , require ('./functions/crypto')\n    , require ('./functions/time')\n    , require ('./functions/throttle')\n    , require ('./functions/misc')\n))\n\n/*  ------------------------------------------------------------------------ */\n","\"use strict\";\n\n/*  ------------------------------------------------------------------------ */\n\nconst CryptoJS = require ('crypto-js')\nconst { capitalize } = require ('./string')\nconst { stringToBase64, utf16ToBase64, urlencodeBase64 } = require ('./encode')\n\n/*  ------------------------------------------------------------------------ */\n\nconst hash = (request, hash = 'md5', digest = 'hex') => {\n    const result = CryptoJS[hash.toUpperCase ()] (request)\n    return (digest == 'binary') ? result : result.toString (CryptoJS.enc[capitalize (digest)])\n}\n\n/*  .............................................   */\n\nconst hmac = (request, secret, hash = 'sha256', digest = 'hex') => {\n    const encoding = (digest == 'binary') ? 'Latin1' : capitalize (digest)\n    return CryptoJS['Hmac' + hash.toUpperCase ()] (request, secret).toString (CryptoJS.enc[capitalize (encoding)])\n}\n\n/*  .............................................   */\n\nconst jwt = function JSON_web_token (request, secret, alg = 'HS256', hash = 'sha256') {\n    const encodedHeader = urlencodeBase64 (stringToBase64 (JSON.stringify ({ 'alg': alg, 'typ': 'JWT' })))\n        , encodedData = urlencodeBase64 (stringToBase64 (JSON.stringify (request)))\n        , token = [ encodedHeader, encodedData ].join ('.')\n        , signature = urlencodeBase64 (utf16ToBase64 (hmac (token, secret, hash, 'utf16')))\n    return [ token, signature ].join ('.')\n}\n\n/*  ------------------------------------------------------------------------ */\n\nmodule.exports = {\n\n    hash,\n    hmac,\n    jwt\n}\n\n/*  ------------------------------------------------------------------------ */\n","\"use strict\";\n\n/*  ------------------------------------------------------------------------ */\n\nconst CryptoJS = require ('crypto-js')\nconst qs       = require ('qs') // querystring (TODO: get rid of that dependency)\n\n/*  ------------------------------------------------------------------------ */\n\nmodule.exports =\n    \n    { json:   JSON.stringify\n    , unjson: JSON.parse\n\n    , stringToBinary (str) {\n        const arr = new Uint8Array (str.length)\n        for (let i = 0; i < str.length; i++) { arr[i] = str.charCodeAt (i); }\n        return CryptoJS.lib.WordArray.create (arr)\n    }\n\n    , stringToBase64: string => CryptoJS.enc.Latin1.parse (string).toString (CryptoJS.enc.Base64)\n    , utf16ToBase64:  string => CryptoJS.enc.Utf16 .parse (string).toString (CryptoJS.enc.Base64)\n    , base64ToBinary: string => CryptoJS.enc.Base64.parse (string)\n    , base64ToString: string => CryptoJS.enc.Base64.parse (string).toString (CryptoJS.enc.Utf8)\n    , binaryToString: string => string\n\n    , binaryConcat: (...args) => args.reduce ((a, b) => a.concat (b))\n\n    , urlencode: object => qs.stringify (object)\n    , rawencode: object => qs.stringify (object, { encode: false })\n\n    // Url-safe-base64 without equals signs, with + replaced by - and slashes replaced by underscores\n\n    , urlencodeBase64: base64string => base64string.replace (/[=]+$/, '')\n                                                   .replace (/\\+/g, '-')\n                                                   .replace (/\\//g, '_')\n}\n\n/*  ------------------------------------------------------------------------ */\n","\"use strict\";\n\n/*  ------------------------------------------------------------------------ */\n\nconst { isObject, isNumber, isDictionary, isArray } = require ('./type')\n\n/*  ------------------------------------------------------------------------ */\n\nconst empty = () => Object.create (null) // empty obj without even a prototype\n\n    , keys = Object.keys\n\n    , values = x => !isArray (x)  // don't copy arrays if they're already arrays!\n                        ? Object.values (x)\n                        : x\n\n    , index = x => new Set (values (x))\n\n    , extend = (...args) => Object.assign (empty (), ...args) // NB: side-effect free\n\n    , clone = x => isArray (x)\n                        ? Array.from (x) // clones arrays\n                        : extend (x)     // clones objects\n\n/*  ------------------------------------------------------------------------ */\n\nmodule.exports = {\n\n      empty\n    , keys\n    , values\n    , extend\n    , clone\n    , index\n    , ordered: x => x // a stub to keep assoc keys in order (in JS it does nothing, it's mostly for Python)   \n    , unique:  x => Array.from (index (x))\n    \n/*  .............................................   */\n\n    , keysort (x, out = empty ()) {\n        \n        for (const k of keys (x).sort ())\n            out[k] = x[k]\n        \n        return out\n    }\n\n/*  .............................................   */\n\n    , indexBy (x, k, out = empty ()) {\n\n        for (const v of values (x))\n            if (k in v)\n                out[v[k]] = v\n    \n        return out\n    }\n\n/*  .............................................   */\n\n    , groupBy (x, k, out = empty ()) {\n\n        for (const v of values (x)) {\n            if (k in v) {\n                const p = v[k]\n                out[p] = out[p] || []\n                out[p].push (v)\n            }\n        }\n        return out\n    }\n\n/*  .............................................   */\n\n    , filterBy (x, k, value = undefined, out = []) {\n\n        for (const v of values (x))\n            if (v[k] === value)\n                out.push (v)\n    \n        return out\n    }\n\n/*  .............................................   */\n\n    , sortBy: (array, // NB: MUTATES ARRAY!\n               key,\n               descending = false,\n               direction  = descending ? -1 : 1) => array.sort ((a, b) =>\n                                                                ((a[key] < b[key]) ? -direction :\n                                                                ((a[key] > b[key]) ?  direction : 0)))\n\n/*  .............................................   */\n\n    , flatten: function flatten (x, out = []) {\n\n        for (const v of x) {\n            if (isArray (v)) flatten (v, out)\n            else out.push (v)\n        }\n    \n        return out\n    }\n\n/*  .............................................   */\n\n    , pluck: (x, k) => values (x)\n                        .filter (v => k in v)\n                        .map (v => v[k])\n\n/*  .............................................   */\n\n    , omit (x, ...args) {\n\n        const out = clone (x)\n    \n        for (const k of args) {\n    \n            if (isArray (k)) // omit (x, ['a', 'b'])\n                for (const kk of k)\n                    delete out[kk]\n\n            else delete out[k] // omit (x, 'a', 'b')\n        }\n        \n        return out\n    }\n\n/*  .............................................   */\n\n    , sum (...xs) {\n\n        const ns = xs.filter (isNumber) // leave only numbers\n    \n        return (ns.length > 0)\n                    ? ns.reduce ((a, b) => a + b, 0)\n                    : undefined\n    }\n\n/*  .............................................   */\n\n    , deepExtend: function deepExtend (...xs) {\n\n        let out = undefined\n\n        for (const x of xs) {\n\n            if (isDictionary (x)) {\n\n                if (!isObject (out))\n                    out = empty ()\n\n                for (const k in x)\n                    out[k] = deepExtend (out[k], x[k])\n\n            } else out = x\n        }\n\n        return out\n    }\n\n/*  ------------------------------------------------------------------------ */\n\n}\n","\"use strict\";\n\n/*  ------------------------------------------------------------------------ */\n\nmodule.exports = {\n    \n    aggregate (bidasks) { \n\n        let result = {}\n\n        for (const [price, volume] of bidasks) {\n            if (volume > 0)\n                result[price] = (result[price] || 0) + volume\n        }\n\n        return Object.keys (result)\n                    .map (price => [parseFloat (price),\n                                    parseFloat (result[price])])\n    }\n}\n\n/*  ------------------------------------------------------------------------ */\n","/*  NB: A LEGACY CODE, WILL BE RE-WRITTEN VERY SOON\n    ------------------------------------------------------------------------ */\n\n// See https://stackoverflow.com/questions/1685680/how-to-avoid-scientific-notation-for-large-numbers-in-javascript for discussion\n\nfunction toFixed (x) { // avoid scientific notation for too large and too small numbers\n\n    if (Math.abs (x) < 1.0) {\n        const e = parseInt (x.toString ().split ('e-')[1])\n        if (e) {\n            x *= Math.pow (10, e-1)\n            x = '0.' + (new Array (e)).join ('0') + x.toString ().substring (2)\n        }\n    } else {\n        let e = parseInt (x.toString ().split ('+')[1])\n        if (e > 20) {\n            e -= 20\n            x /= Math.pow (10, e)\n            x += (new Array (e+1)).join ('0')\n        }\n    }\n    return x\n}\n\n// See https://stackoverflow.com/questions/4912788/truncate-not-round-off-decimal-numbers-in-javascript for discussion\n\n// > So, after all it turned out, rounding bugs will always haunt you, no matter how hard you try to compensate them.\n// > Hence the problem should be attacked by representing numbers exactly in decimal notation.\n\nconst truncate_regExpCache = []\n    , truncate_to_string = (num, precision = 0) => {\n        num = toFixed (num)\n        if (precision > 0) {\n            const re = truncate_regExpCache[precision] || (truncate_regExpCache[precision] = new RegExp(\"([-]*\\\\d+\\\\.\\\\d{\" + precision + \"})(\\\\d)\"))\n            const [,result] = num.toString ().match (re) || [null, num]\n            return result.toString ()\n        }\n        return parseInt (num).toString ()\n    }\n    , truncate = (num, precision = 0) => parseFloat (truncate_to_string (num, precision))\n\nconst precisionFromString = (string) => {\n    const split = string.replace (/0+$/g, '').split ('.')\n    return (split.length > 1) ? (split[1].length) : 0\n}\n\nmodule.exports = {\n\n    toFixed,\n    truncate_to_string,\n    truncate,\n    precisionFromString\n}\n\n// \"use strict\";\n\n// const { isString, isNumber } = require ('./type')\n\n// /*  ------------------------------------------------------------------------ */\n\n// const decimal = float => parseFloat (float).toString ()\n\n// /*  ------------------------------------------------------------------------ */\n\n// // See https://stackoverflow.com/questions/1685680/how-to-avoid-scientific-notation-for-large-numbers-in-javascript for discussion\n\n// function numberToString (x) { // avoid scientific notation for too large and too small numbers\n\n//     if (isString (x)) return x\n\n//     if (Math.abs (x) < 1.0) {\n//         const e = parseInt (x.toString ().split ('e-')[1])\n//         if (e) {\n//             x *= Math.pow (10, e-1)\n//             x = '0.' + (new Array (e)).join ('0') + x.toString ().substring (2)\n//         }\n//     } else {\n//         let e = parseInt (x.toString ().split ('+')[1])\n//         if (e > 20) {\n//             e -= 20\n//             x /= Math.pow (10, e)\n//             x += (new Array (e+1)).join ('0')\n//         }\n//     }\n//     return x.toString ()\n// }\n\n// /*  ------------------------------------------------------------------------ */\n\n// const padWithZeroes = (x, digits = 0) => {\n\n//     const [int, frac = ''] = x.split ('.')\n\n//     return int + ((frac || (digits > 0))\n//                         ? ('.' + frac.padEnd (digits, '0'))\n//                         : '')\n// }\n\n// /*  ------------------------------------------------------------------------ */\n\n// const roundDecimalString = (s, to, afterDot = false) => { \n\n//     const digits = Array.from (s)\n//     const result = []\n//     const dot = s.indexOf ('.')\n\n//     let memo = 0\n\n//     if (afterDot) to = ((dot >= 0) ? dot : digits.length) + to\n\n//     for (let i = digits.length - 1; i >= 0; i--) {\n//         const d = digits[i]\n//         if (d !== '.') {\n//             let n = (d.charCodeAt (0) - 48) + memo\n//             let numDigitsAhead = i\n//             let dotAhead = (dot >= 0) && (i >= dot)\n//             if ((numDigitsAhead + (dotAhead ? -1 : 0)) >= to) { // ignore dot when counting digits ahead\n//                 n = (n > 5) ? 10 : 0 // rounding on per-digit basis\n//             }\n//             if (n > 9) { n = 0; memo = 1; }\n//             else memo = 0\n//             digits[i] = n\n//         }\n//     }\n//     return (memo || '') + digits.join ('')\n// }\n\n// /*  ------------------------------------------------------------------------ */\n\n// const roundNumber = (x, { digits = 8, fixed = true }) => { // accepts either strings or Numbers\n    \n//     const s = numberToString (x)\n\n//     if (fixed) {\n//         return roundDecimalString (s, digits, true)\n\n//     } else {\n//         const [,zeros,significantPart] = s.match (/^([^1-9]*)(.+)$/)\n//         return zeros + roundDecimalString (significantPart, digits)\n//     }\n// }\n\n// /*  ------------------------------------------------------------------------ */\n\n// // See https://stackoverflow.com/questions/4912788/truncate-not-round-off-decimal-numbers-in-javascript for discussion\n\n// // > So, after all it turned out, rounding bugs will always haunt you, no matter how hard you try to compensate them.\n// // > Hence the problem should be attacked by representing numbers exactly in decimal notation.\n\n// const regexCache = []\n// const truncNumber = (x, { digits = 0, fixed = true }) => { // accepts either strings or Numbers\n\n//     const s = numberToString (x)\n\n//     if (digits > 0) {\n//         const re = regexCache[digits] || (regexCache[digits] = new RegExp(\"([-]*\\\\d+\\\\.\\\\d{\" + digits + \"})(\\\\d)\"))\n//         const [,result] = s.match (re) || [null, s]\n//         return fixed\n//                 ? padWithZeroes (result, digits)\n//                 : result\n\n//     } else {\n//         throw new Error ('not implemented yet')\n//     }\n// }\n\n// /*  ------------------------------------------------------------------------ */\n\n// const precisionFromString = (string) => {\n//     const split = string.replace (/0+$/g, '').split ('.')\n//     return (split.length > 1) ? (split[1].length) : 0\n// }\n\n// /*  ------------------------------------------------------------------------ */\n\n// const toPrecision = (x, { round = true, digits = 8, fixed = true }) => round ? roundNumber (x, { digits, fixed })\n//                                                                              : truncNumber (x, { digits, fixed })\n\n// /*  ------------------------------------------------------------------------ */\n\n// module.exports = {\n \n//     decimal,\n//     numberToString,\n//     toPrecision,\n//     padWithZeroes,\n//     roundDecimalString,\n//     precisionFromString\n// }\n\n// /*  ------------------------------------------------------------------------ */\n","\"use strict\";\n\n/*  ------------------------------------------------------------------------ */\n\nmodule.exports = {\n\n    isNode: (typeof window === 'undefined') &&\n          !((typeof WorkerGlobalScope !== 'undefined') && (self instanceof WorkerGlobalScope))\n}\n\n/*  ------------------------------------------------------------------------ */\n","\"use strict\";\n\n/*  ------------------------------------------------------------------------ */\n\nmodule.exports =\n    \n    { uuid: a => a ? (a ^ Math.random () * 16 >> a / 4).toString (16)\n                   : ([1e7]+-1e3+-4e3+-8e3+-1e11).replace (/[018]/g, uuid)\n        \n    , unCamelCase: s => s.replace (/[a-z0-9][A-Z]/g, x => x[0] + '_' + x[1]).toLowerCase () // hasFetchOHLCV → has_fetch_ohlcv\n\n    , capitalize: s => s.length\n                            ? (s.charAt (0).toUpperCase () + s.slice (1))\n                            : s\n    }\n\n/*  ------------------------------------------------------------------------ */\n","\"use strict\";\n\n/*  ------------------------------------------------------------------------ */\n\nconst { sleep\n      , now } = require ('./time')\n\n/*  ------------------------------------------------------------------------ */\n\nmodule.exports = {\n    \n    throttle: function throttle (cfg) {\n\n        let   lastTimestamp = now ()\n            , numTokens     = (typeof cfg.numTokens != 'undefined') ? cfg.numTokens : cfg.capacity\n            , running       = false\n            , counter       = 0\n\n        const queue = []\n\n        return Object.assign (cost => {\n\n            if (queue.length > cfg.maxCapacity)\n                throw new Error ('Backlog is over max capacity of ' + cfg.maxCapacity)\n\n            return new Promise (async (resolve, reject) => {\n\n                try {\n                    queue.push ({ cost, resolve, reject })\n\n                    if (!running) {\n                        running = true\n                        while (queue.length > 0) {\n                            const hasEnoughTokens = cfg.capacity ? (numTokens > 0) : (numTokens >= 0)\n                            if (hasEnoughTokens) {\n                                if (queue.length > 0) {\n                                    let { cost, resolve, reject } = queue[0]\n                                    cost = (cost || cfg.defaultCost)\n                                    if (numTokens >= Math.min (cost, cfg.capacity)) {\n                                        numTokens -= cost\n                                        queue.shift ()\n                                        resolve ()\n                                    }\n                                }\n                            }\n                            const t = now ()\n                                , elapsed = t - lastTimestamp\n                            lastTimestamp = t\n                            numTokens = Math.min (cfg.capacity, numTokens + elapsed * cfg.refillRate)\n                            await sleep (cfg.delay)\n                        }\n                        running = false\n                    }\n\n                } catch (e) {\n                    reject (e)\n                }\n            })\n\n        }, cfg, { configure: newCfg => throttle (Object.assign ({}, cfg, newCfg)) })\n    }\n}\n\n/*  ------------------------------------------------------------------------ */\n","\"use strict\";\n\n/*  ------------------------------------------------------------------------ */\n\nconst { isNode } = require ('./platform')\n\n/*  ------------------------------------------------------------------------ */\n\nconst now = Date.now // TODO: figure out how to utilize performance.now () properly – it's not as easy as it does not return a unix timestamp...\n\n/*  ------------------------------------------------------------------------ */\n\nconst setTimeout_original = setTimeout\nconst setTimeout_safe = (done, ms, setTimeout = setTimeout_original /* overrideable for mocking purposes */, targetTime = now () + ms) => {\n\n/*  The built-in setTimeout function can fire its callback earlier than specified, so we\n    need to ensure that it does not happen: sleep recursively until `targetTime` is reached...   */\n\n    let clearInnerTimeout = () => {}\n    let active = true\n\n    let id = setTimeout (() => {\n        active = true\n        const rest = targetTime - now ()\n        if (rest > 0) {\n            clearInnerTimeout = setTimeout_safe (done, rest, setTimeout, targetTime) // try sleep more\n        } else {\n            done ()\n        }\n    }, ms)\n\n    return function clear () {\n        if (active) {\n            active = false // dunno if IDs are unique on various platforms, so it's better to rely on this flag to exclude the possible cancellation of the wrong timer (if called after completion)\n            clearTimeout (id)\n        }\n        clearInnerTimeout ()\n    }\n}\n\n/*  ------------------------------------------------------------------------ */\n\nclass TimedOut extends Error {\n\n    constructor () {\n        const message = 'timed out'\n        super (message)\n        this.constructor = TimedOut\n        this.__proto__   = TimedOut.prototype\n        this.message     = message\n    }\n}\n\n/*  ------------------------------------------------------------------------ */\n\nmodule.exports =\n\n    { now\n    , setTimeout_safe\n    , sleep: ms => new Promise (resolve => setTimeout_safe (resolve, ms))\n    , TimedOut\n    , timeout: async (ms, promise) => {\n\n        let clear = () => {}\n        const expires = new Promise (resolve => (clear = setTimeout_safe (resolve, ms)))\n\n        try {\n            return await Promise.race ([promise, expires.then (() => { throw new TimedOut () })])\n        } finally {\n            clear () // fixes https://github.com/ccxt/ccxt/issues/749\n        }\n    }\n}\n\n/*  ------------------------------------------------------------------------ */\n","\"use strict\";\n\n/*  ------------------------------------------------------------------------ */\n\nconst isNumber          = Number.isFinite\n    , isArray           = Array.isArray\n    , isString          = s =>                 (typeof s === 'string')\n    , isObject          = o => (o !== null) && (typeof o === 'object')\n    , isDictionary      = o => (isObject (o) && !isArray (o))\n    , isStringCoercible = x => (hasProps (x) && x.toString) || isNumber (x)\n\n/*  .............................................   */\n\nconst hasProps = o => (o !== undefined) &&\n                      (o !== null)\n\n    , prop = (o, k) => isObject (o) ? o[k]\n                                    : undefined\n\n/*  .............................................   */\n\nconst asFloat   = x => (isNumber (x) || isString (x)) ? parseFloat (x)     : NaN\n    , asInteger = x => (isNumber (x) || isString (x)) ? parseInt   (x, 10) : NaN\n\n/*  .............................................   */\n\nmodule.exports =\n\n    { isNumber\n    , isArray\n    , isObject\n    , isString\n    , isStringCoercible\n    , isDictionary\n\n    , hasProps\n    , prop\n\n    , asFloat\n    , asInteger\n    \n    , safeFloat:   (o, k, $default, n =   asFloat (prop (o, k))) => isNumber (n)          ? n          : $default\n    , safeInteger: (o, k, $default, n = asInteger (prop (o, k))) => isNumber (n)          ? n          : $default\n    , safeValue:   (o, k, $default, x =            prop (o, k) ) => hasProps (x)          ? x          : $default\n    , safeString:  (o, k, $default, x =            prop (o, k) ) => isStringCoercible (x) ? String (x) : $default\n\n    }\n\n/*  ------------------------------------------------------------------------ */\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, AuthenticationError, DDoSProtection } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bibox extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bibox',\n            'name': 'Bibox',\n            'countries': [ 'CN', 'US', 'KR' ],\n            'version': 'v1',\n            'has': {\n                'CORS': false,\n                'publicAPI': false,\n                'fetchBalance': true,\n                'fetchCurrencies': true,\n                'fetchTickers': true,\n                'fetchOrders': true,\n                'fetchMyTrades': true,\n                'fetchOHLCV': true,\n                'withdraw': true,\n            },\n            'timeframes': {\n                '1m': '1min',\n                '5m': '5min',\n                '15m': '15min',\n                '30m': '30min',\n                '1h': '1hour',\n                '8h': '12hour',\n                '1d': 'day',\n                '1w': 'week',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/34902611-2be8bf1a-f830-11e7-91a2-11b2f292e750.jpg',\n                'api': 'https://api.bibox.com',\n                'www': 'https://www.bibox.com',\n                'doc': 'https://github.com/Biboxcom/api_reference/wiki/home_en',\n                'fees': 'https://bibox.zendesk.com/hc/en-us/articles/115004417013-Fee-Structure-on-Bibox',\n            },\n            'api': {\n                'public': {\n                    'post': [\n                        // TODO: rework for full endpoint/cmd paths here\n                        'mdata',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'user',\n                        'orderpending',\n                        'transfer',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'taker': 0.001,\n                    'maker': 0.0,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                    },\n                    'deposit': 0.0,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicPostMdata ({\n            'cmd': 'api/marketAll',\n            'body': {},\n        });\n        let markets = response['result'];\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let base = market['coin_symbol'];\n            let quote = market['currency_symbol'];\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            let id = base + '_' + quote;\n            let precision = {\n                'amount': 8,\n                'price': 8,\n            };\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': undefined,\n                'info': market,\n                'lot': Math.pow (10, -precision['amount']),\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision['amount']),\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                },\n            }));\n        }\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.safeInteger (ticker, 'timestamp', this.seconds ());\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else {\n            symbol = ticker['coin_symbol'] + '/' + ticker['currency_symbol'];\n        }\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'buy'),\n            'ask': this.safeFloat (ticker, 'sell'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': this.safeString (ticker, 'percent'),\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'vol'),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicPostMdata ({\n            'cmd': 'api/ticker',\n            'body': this.extend ({\n                'pair': market['id'],\n            }, params),\n        });\n        return this.parseTicker (response['result'], market);\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        let response = await this.publicPostMdata ({\n            'cmd': 'api/marketAll',\n            'body': {},\n        });\n        let tickers = response['result'];\n        let result = {};\n        for (let t = 0; t < tickers.length; t++) {\n            let ticker = this.parseTicker (tickers[t]);\n            let symbol = ticker['symbol'];\n            if (symbols && (!(symbol in symbols))) {\n                continue;\n            }\n            result[symbol] = ticker;\n        }\n        return result;\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = trade['time'];\n        let side = (trade['side'] === '1') ? 'buy' : 'sell';\n        return {\n            'id': undefined,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': 'limit',\n            'side': side,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let size = (limit) ? limit : 200;\n        let response = await this.publicPostMdata ({\n            'cmd': 'api/deals',\n            'body': this.extend ({\n                'pair': market['id'],\n                'size': size,\n            }, params),\n        });\n        return this.parseTrades (response['result'], market, since, limit);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicPostMdata ({\n            'cmd': 'api/depth',\n            'body': this.extend ({\n                'pair': market['id'],\n            }, params),\n        });\n        return this.parseOrderBook (response['result'], this.safeFloat (response['result'], 'update_time'), 'bids', 'asks', 'price', 'amount');\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv['time'],\n            ohlcv['open'],\n            ohlcv['high'],\n            ohlcv['low'],\n            ohlcv['close'],\n            ohlcv['vol'],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let size = (limit) ? limit : 1000;\n        let response = await this.publicPostMdata ({\n            'cmd': 'api/kline',\n            'body': this.extend ({\n                'pair': market['id'],\n                'period': this.timeframes[timeframe],\n                'size': size,\n            }, params),\n        });\n        return this.parseOHLCVs (response['result'], market, timeframe, since, limit);\n    }\n\n    async fetchCurrencies (params = {}) {\n        let response = await this.privatePostTransfer ({\n            'cmd': 'transfer/coinList',\n            'body': {},\n        });\n        let currencies = response['result'];\n        let result = {};\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let id = currency['symbol'];\n            let code = this.commonCurrencyCode (id);\n            let precision = 8;\n            let deposit = currency['enable_deposit'];\n            let withdraw = currency['enable_withdraw'];\n            let active = (deposit && withdraw);\n            result[code] = {\n                'id': id,\n                'code': code,\n                'info': currency,\n                'name': currency['name'],\n                'active': active,\n                'status': 'ok',\n                'fee': undefined,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': undefined,\n                        'max': Math.pow (10, precision),\n                    },\n                },\n            };\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostTransfer ({\n            'cmd': 'transfer/assets',\n            'body': this.extend ({\n                'select': 1,\n            }, params),\n        });\n        let balances = response['result'];\n        let result = { 'info': balances };\n        let indexed = undefined;\n        if ('assets_list' in balances) {\n            indexed = this.indexBy (balances['assets_list'], 'coin_symbol');\n        } else {\n            indexed = {};\n        }\n        let keys = Object.keys (indexed);\n        for (let i = 0; i < keys.length; i++) {\n            let id = keys[i];\n            let currency = this.commonCurrencyCode (id);\n            let account = this.account ();\n            let balance = indexed[id];\n            let used = parseFloat (balance['freeze']);\n            let free = parseFloat (balance['balance']);\n            let total = this.sum (free, used);\n            account['free'] = free;\n            account['used'] = used;\n            account['total'] = total;\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orderType = (type === 'limit') ? 2 : 1;\n        let response = await this.privatePostOrder ({\n            'cmd': 'orderpending/trade',\n            'body': this.extend ({\n                'pair': market['id'],\n                'account_type': 0,\n                'order_type': orderType,\n                'order_side': side,\n                'pay_bix': 0,\n                'amount': amount,\n                'price': price,\n            }, params),\n        });\n        return {\n            'info': response,\n            'id': this.safeString (response, 'result'),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        let response = await this.privatePostCancelOrder ({\n            'cmd': 'orderpending/cancelTrade',\n            'body': this.extend ({\n                'orders_id': id,\n            }, params),\n        });\n        return response;\n    }\n\n    parseOrder (order, market = undefined) {\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else {\n            symbol = order['coin_symbol'] + '/' + order['currency_symbol'];\n        }\n        let type = (order['order_type'] === 1) ? 'market' : 'limit';\n        let timestamp = order['createdAt'];\n        let price = order['price'];\n        let filled = order['amount'];\n        let amount = this.safeInteger (order, 'deal_amount');\n        let remaining = amount - filled;\n        let side = (order['order_side'] === 1) ? 'buy' : 'sell';\n        let status = undefined;\n        if ('status' in order) {\n            status = this.parseOrderStatus (order['status']);\n        }\n        let result = {\n            'info': order,\n            'id': this.safeString (order, 'id'),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': price,\n            'amount': amount,\n            'cost': price * filled,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': this.safeFloat (order, 'fee'),\n        };\n        return result;\n    }\n\n    parseOrderStatus (status) {\n        let statuses = {\n            '1': 'pending',\n            '2': 'open',\n            '3': 'closed',\n            '4': 'canceled',\n            '5': 'canceled',\n            '6': 'canceled',\n        };\n        return this.safeString (statuses, status, status.toLowerCase ());\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrders requires a symbol param');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let size = (limit) ? limit : 200;\n        let response = await this.privatePostOrderpending ({\n            'cmd': 'orderpending/orderPendingList',\n            'body': this.extend ({\n                'pair': market['id'],\n                'account_type': 0, // 0 - regular, 1 - margin\n                'page': 1,\n                'size': size,\n            }, params),\n        });\n        let orders = ('items' in response) ? response['items'] : [];\n        return this.parseOrders (orders, market, since, limit);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchMyTrades requires a symbol param');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let size = (limit) ? limit : 200;\n        let response = await this.privatePostOrderpending ({\n            'cmd': 'orderpending/orderHistoryList',\n            'body': this.extend ({\n                'pair': market['id'],\n                'account_type': 0, // 0 - regular, 1 - margin\n                'page': 1,\n                'size': size,\n            }, params),\n        });\n        let orders = ('items' in response) ? response['items'] : [];\n        return this.parseOrders (orders, market, since, limit);\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (currency);\n        let response = await this.privatePostTransfer ({\n            'cmd': 'transfer/transferOutInfo',\n            'body': this.extend ({\n                'coin_symbol': market['id'],\n            }, params),\n        });\n        let result = {\n            'info': response,\n            'address': undefined,\n        };\n        return result;\n    }\n\n    async withdraw (code, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let currency = this.currency (code);\n        let response = await this.privatePostTransfer ({\n            'cmd': 'transfer/transferOut',\n            'body': this.extend ({\n                'coin_symbol': currency,\n                'amount': amount,\n                'addr': address,\n                'addr_remark': '',\n            }, params),\n        });\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + path;\n        let cmds = this.json ([ params ]);\n        if (api === 'public') {\n            body = {\n                'cmds': cmds,\n            };\n        } else {\n            this.checkRequiredCredentials ();\n            body = {\n                'cmds': cmds,\n                'apikey': this.apiKey,\n                'sign': this.hmac (this.encode (cmds), this.encode (this.secret), 'md5'),\n            };\n        }\n        headers = { 'Content-Type': 'application/json' };\n        return { 'url': url, 'method': method, 'body': this.json (body), 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        let message = this.id + ' ' + this.json (response);\n        if ('error' in response) {\n            if ('code' in response['error']) {\n                let code = response['error']['code'];\n                if (code === '3012')\n                    throw new AuthenticationError (message); // invalid api key\n                else if (code === '3025')\n                    throw new AuthenticationError (message); // signature failed\n                else if (code === '4003')\n                    throw new DDoSProtection (message); // server is busy, try again later\n            }\n            throw new ExchangeError (message);\n        }\n        if (!('result' in response))\n            throw new ExchangeError (message);\n        return response['result'][0];\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InsufficientFunds, OrderNotFound, InvalidOrder, DDoSProtection } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class binance extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'binance',\n            'name': 'Binance',\n            'countries': 'JP', // Japan\n            'rateLimit': 500,\n            // new metainfo interface\n            'has': {\n                'CORS': false,\n                'fetchBidsAsks': true,\n                'fetchTickers': true,\n                'fetchOHLCV': true,\n                'fetchMyTrades': true,\n                'fetchOrder': true,\n                'fetchOrders': true,\n                'fetchOpenOrders': true,\n                'withdraw': true,\n            },\n            'timeframes': {\n                '1m': '1m',\n                '3m': '3m',\n                '5m': '5m',\n                '15m': '15m',\n                '30m': '30m',\n                '1h': '1h',\n                '2h': '2h',\n                '4h': '4h',\n                '6h': '6h',\n                '8h': '8h',\n                '12h': '12h',\n                '1d': '1d',\n                '3d': '3d',\n                '1w': '1w',\n                '1M': '1M',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/29604020-d5483cdc-87ee-11e7-94c7-d1a8d9169293.jpg',\n                'api': {\n                    'web': 'https://www.binance.com',\n                    'wapi': 'https://api.binance.com/wapi/v3',\n                    'public': 'https://api.binance.com/api/v1',\n                    'private': 'https://api.binance.com/api/v3',\n                    'v3': 'https://api.binance.com/api/v3',\n                    'v1': 'https://api.binance.com/api/v1',\n                },\n                'www': 'https://www.binance.com',\n                'doc': 'https://github.com/binance-exchange/binance-official-api-docs/blob/master/rest-api.md',\n                'fees': [\n                    'https://binance.zendesk.com/hc/en-us/articles/115000429332',\n                    'https://support.binance.com/hc/en-us/articles/115000583311',\n                ],\n            },\n            'api': {\n                'web': {\n                    'get': [\n                        'exchange/public/product',\n                    ],\n                },\n                'wapi': {\n                    'post': [\n                        'withdraw',\n                    ],\n                    'get': [\n                        'depositHistory',\n                        'withdrawHistory',\n                        'depositAddress',\n                    ],\n                },\n                'v3': {\n                    'get': [\n                        'ticker/price',\n                        'ticker/bookTicker',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'exchangeInfo',\n                        'ping',\n                        'time',\n                        'depth',\n                        'aggTrades',\n                        'klines',\n                        'ticker/24hr',\n                        'ticker/allPrices',\n                        'ticker/allBookTickers',\n                        'ticker/price',\n                        'ticker/bookTicker',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'order',\n                        'openOrders',\n                        'allOrders',\n                        'account',\n                        'myTrades',\n                    ],\n                    'post': [\n                        'order',\n                        'order/test',\n                    ],\n                    'delete': [\n                        'order',\n                    ],\n                },\n                'v1': {\n                    'put': [ 'userDataStream' ],\n                    'post': [ 'userDataStream' ],\n                    'delete': [ 'userDataStream' ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'taker': 0.001,\n                    'maker': 0.001,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BNB': 1.0,\n                        'BTC': 0.001,\n                        'ETH': 0.01,\n                        'LTC': 0.01,\n                        'NEO': 0.0,\n                        'QTUM': 0.01,\n                        'SNT': 10.0,\n                        'BNT': 1.2,\n                        'EOS': 0.7,\n                        'BCH': 0.0005,\n                        'GAS': 0.0,\n                        'USDT': 25.0,\n                        'OAX': 6.0,\n                        'DNT': 60.0,\n                        'MCO': 0.3,\n                        'ICN': 2.0,\n                        'WTC': 0.4,\n                        'OMG': 0.3,\n                        'ZRX': 10.0,\n                        'STRAT': 0.1,\n                        'SNGLS': 20.0,\n                        'BQX': 2.0,\n                        'KNC': 2.0,\n                        'FUN': 80.0,\n                        'SNM': 20.0,\n                        'LINK': 10.0,\n                        'XVG': 0.1,\n                        'CTR': 7.0,\n                        'SALT': 0.4,\n                        'IOTA': 0.5,\n                        'MDA': 2.0,\n                        'MTL': 0.5,\n                        'SUB': 4.0,\n                        'ETC': 0.01,\n                        'MTH': 35.0,\n                        'ENG': 5.0,\n                        'AST': 10.0,\n                        'BTG': undefined,\n                        'DASH': 0.002,\n                        'EVX': 2.5,\n                        'REQ': 15.0,\n                        'LRC': 12.0,\n                        'VIB': 20.0,\n                        'HSR': 0.0001,\n                        'TRX': 30.0,\n                        'POWR': 5.0,\n                        'ARK': 0.1,\n                        'YOYO': 10.0,\n                        'XRP': 0.15,\n                        'MOD': 2.0,\n                        'ENJ': 80.0,\n                        'STORJ': 3.0,\n                        'VEN': 5.0,\n                        'KMD': 1.0,\n                        'NULS': 4.0,\n                        'RCN': 20.0,\n                        'RDN': 0.3,\n                        'XMR': 0.04,\n                        'DLT': 15.0,\n                        'AMB': 10.0,\n                        'BAT': 15.0,\n                        'ZEC': 0.005,\n                        'BCPT': 14.0,\n                        'ARN': 7.0,\n                        'GVT': 0.5,\n                        'CDT': 35.0,\n                        'GXS': 0.3,\n                        'POE': 50.0,\n                        'QSP': 30.0,\n                        'BTS': 1.0,\n                        'XZC': 0.02,\n                        'LSK': 0.1,\n                        'TNT': 35.0,\n                        'FUEL': 60.0,\n                        'MANA': 30.0,\n                        'BCD': 0.0005,\n                        'DGD': 0.03,\n                        'ADX': 2.0,\n                        'ADA': 1.0,\n                        'PPT': 0.1,\n                        'CMT': 15.0,\n                        'XLM': 0.01,\n                        'CND': 180.0,\n                        'LEND': 50.0,\n                        'WABI': 4.0,\n                        'TNB': 70.0,\n                        'WAVES': 0.002,\n                        'ICX': 1.5,\n                        'GTO': 30.0,\n                        'OST': 15.0,\n                        'ELF': 2.0,\n                        'AION': 1.0,\n                        'NEBL': 0.01,\n                        'BRD': 3.0,\n                        'EDO': 1.5,\n                        'WINGS': 3.0,\n                        'NAV': 0.2,\n                        'LUN': 0.3,\n                        'TRIG': 5.0,\n                    },\n                    'deposit': {\n                        'BNB': 0,\n                        'BTC': 0,\n                        'ETH': 0,\n                        'LTC': 0,\n                        'NEO': 0,\n                        'QTUM': 0,\n                        'SNT': 0,\n                        'BNT': 0,\n                        'EOS': 0,\n                        'BCH': 0,\n                        'GAS': 0,\n                        'USDT': 0,\n                        'OAX': 0,\n                        'DNT': 0,\n                        'MCO': 0,\n                        'ICN': 0,\n                        'WTC': 0,\n                        'OMG': 0,\n                        'ZRX': 0,\n                        'STRAT': 0,\n                        'SNGLS': 0,\n                        'BQX': 0,\n                        'KNC': 0,\n                        'FUN': 0,\n                        'SNM': 0,\n                        'LINK': 0,\n                        'XVG': 0,\n                        'CTR': 0,\n                        'SALT': 0,\n                        'IOTA': 0,\n                        'MDA': 0,\n                        'MTL': 0,\n                        'SUB': 0,\n                        'ETC': 0,\n                        'MTH': 0,\n                        'ENG': 0,\n                        'AST': 0,\n                        'BTG': 0,\n                        'DASH': 0,\n                        'EVX': 0,\n                        'REQ': 0,\n                        'LRC': 0,\n                        'VIB': 0,\n                        'HSR': 0,\n                        'TRX': 0,\n                        'POWR': 0,\n                        'ARK': 0,\n                        'YOYO': 0,\n                        'XRP': 0,\n                        'MOD': 0,\n                        'ENJ': 0,\n                        'STORJ': 0,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicGetExchangeInfo ();\n        let markets = response['symbols'];\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let id = market['symbol'];\n            if (id === '123456')\n                continue;\n            let baseId = market['baseAsset'];\n            let quoteId = market['quoteAsset'];\n            let base = this.commonCurrencyCode (baseId);\n            let quote = this.commonCurrencyCode (quoteId);\n            let symbol = base + '/' + quote;\n            let filters = this.indexBy (market['filters'], 'filterType');\n            let precision = {\n                'base': market['baseAssetPrecision'],\n                'quote': market['quotePrecision'],\n                'amount': market['baseAssetPrecision'],\n                'price': market['quotePrecision'],\n            };\n            let active = (market['status'] === 'TRADING');\n            let lot = -1 * Math.log10 (precision['amount']);\n            let entry = this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'baseId': baseId,\n                'quoteId': quoteId,\n                'info': market,\n                'lot': lot,\n                'active': active,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': -1 * Math.log10 (precision['price']),\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': lot,\n                        'max': undefined,\n                    },\n                },\n            });\n            if ('PRICE_FILTER' in filters) {\n                let filter = filters['PRICE_FILTER'];\n                entry['precision']['price'] = this.precisionFromString (filter['tickSize']);\n                entry['limits']['price'] = {\n                    'min': parseFloat (filter['minPrice']),\n                    'max': parseFloat (filter['maxPrice']),\n                };\n            }\n            if ('LOT_SIZE' in filters) {\n                let filter = filters['LOT_SIZE'];\n                entry['precision']['amount'] = this.precisionFromString (filter['stepSize']);\n                entry['lot'] = parseFloat (filter['stepSize']);\n                entry['limits']['amount'] = {\n                    'min': parseFloat (filter['minQty']),\n                    'max': parseFloat (filter['maxQty']),\n                };\n            }\n            if ('MIN_NOTIONAL' in filters) {\n                entry['limits']['cost']['min'] = parseFloat (filters['MIN_NOTIONAL']['minNotional']);\n            }\n            result.push (entry);\n        }\n        return result;\n    }\n\n    calculateFee (symbol, type, side, amount, price, takerOrMaker = 'taker', params = {}) {\n        let market = this.markets[symbol];\n        let key = 'quote';\n        let rate = market[takerOrMaker];\n        let cost = parseFloat (this.costToPrecision (symbol, amount * rate));\n        if (side === 'sell') {\n            cost *= price;\n        } else {\n            key = 'base';\n        }\n        return {\n            'type': takerOrMaker,\n            'currency': market[key],\n            'rate': rate,\n            'cost': parseFloat (this.feeToPrecision (symbol, cost)),\n        };\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetAccount (params);\n        let result = { 'info': response };\n        let balances = response['balances'];\n        for (let i = 0; i < balances.length; i++) {\n            let balance = balances[i];\n            let asset = balance['asset'];\n            let currency = this.commonCurrencyCode (asset);\n            let account = {\n                'free': parseFloat (balance['free']),\n                'used': parseFloat (balance['locked']),\n                'total': 0.0,\n            };\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orderbook = await this.publicGetDepth (this.extend ({\n            'symbol': market['id'],\n            'limit': 100, // default = maximum = 100\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.safeInteger (ticker, 'closeTime');\n        if (typeof timestamp === 'undefined')\n            timestamp = this.milliseconds ();\n        let symbol = ticker['symbol'];\n        if (!market) {\n            if (symbol in this.markets_by_id) {\n                market = this.markets_by_id[symbol];\n            }\n        }\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'highPrice'),\n            'low': this.safeFloat (ticker, 'lowPrice'),\n            'bid': this.safeFloat (ticker, 'bidPrice'),\n            'bidVolume': this.safeFloat (ticker, 'bidQty'),\n            'ask': this.safeFloat (ticker, 'askPrice'),\n            'askVolume': this.safeFloat (ticker, 'askQty'),\n            'vwap': this.safeFloat (ticker, 'weightedAvgPrice'),\n            'open': this.safeFloat (ticker, 'openPrice'),\n            'close': this.safeFloat (ticker, 'prevClosePrice'),\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'lastPrice'),\n            'change': this.safeFloat (ticker, 'priceChangePercent'),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'volume'),\n            'quoteVolume': this.safeFloat (ticker, 'quoteVolume'),\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTicker24hr (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTicker (response, market);\n    }\n\n    parseTickers (rawTickers, symbols = undefined) {\n        let tickers = [];\n        for (let i = 0; i < rawTickers.length; i++) {\n            tickers.push (this.parseTicker (rawTickers[i]));\n        }\n        let tickersBySymbol = this.indexBy (tickers, 'symbol');\n        // return all of them if no symbols were passed in the first argument\n        if (typeof symbols === 'undefined')\n            return tickersBySymbol;\n        // otherwise filter by symbol\n        let result = {};\n        for (let i = 0; i < symbols.length; i++) {\n            let symbol = symbols[i];\n            if (symbol in tickersBySymbol)\n                result[symbol] = tickersBySymbol[symbol];\n        }\n        return result;\n    }\n\n    async fetchBidAsks (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let rawTickers = await this.publicGetTickerBookTicker (params);\n        return this.parseTickers (rawTickers, symbols);\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let rawTickers = await this.publicGetTicker24hr (params);\n        return this.parseTickers (rawTickers, symbols);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv[0],\n            parseFloat (ohlcv[1]),\n            parseFloat (ohlcv[2]),\n            parseFloat (ohlcv[3]),\n            parseFloat (ohlcv[4]),\n            parseFloat (ohlcv[5]),\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n            'interval': this.timeframes[timeframe],\n        };\n        request['limit'] = (limit) ? limit : 500; // default == max == 500\n        if (since)\n            request['startTime'] = since;\n        let response = await this.publicGetKlines (this.extend (request, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestampField = ('T' in trade) ? 'T' : 'time';\n        let timestamp = trade[timestampField];\n        let priceField = ('p' in trade) ? 'p' : 'price';\n        let price = parseFloat (trade[priceField]);\n        let amountField = ('q' in trade) ? 'q' : 'qty';\n        let amount = parseFloat (trade[amountField]);\n        let idField = ('a' in trade) ? 'a' : 'id';\n        let id = trade[idField].toString ();\n        let side = undefined;\n        let order = undefined;\n        if ('orderId' in trade)\n            order = trade['orderId'].toString ();\n        if ('m' in trade) {\n            side = trade['m'] ? 'sell' : 'buy'; // this is reversed intentionally\n        } else {\n            side = (trade['isBuyer']) ? 'buy' : 'sell'; // this is a true side\n        }\n        let fee = undefined;\n        if ('commission' in trade) {\n            fee = {\n                'cost': parseFloat (trade['commission']),\n                'currency': this.commonCurrencyCode (trade['commissionAsset']),\n            };\n        }\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': id,\n            'order': order,\n            'type': undefined,\n            'side': side,\n            'price': price,\n            'cost': price * amount,\n            'amount': amount,\n            'fee': fee,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n        };\n        if (since) {\n            request['startTime'] = since;\n            request['endTime'] = since + 3600000;\n        }\n        if (limit)\n            request['limit'] = limit;\n        // 'fromId': 123,    // ID to get aggregate trades from INCLUSIVE.\n        // 'startTime': 456, // Timestamp in ms to get aggregate trades from INCLUSIVE.\n        // 'endTime': 789,   // Timestamp in ms to get aggregate trades until INCLUSIVE.\n        // 'limit': 500,     // default = maximum = 500\n        let response = await this.publicGetAggTrades (this.extend (request, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    parseOrderStatus (status) {\n        if (status === 'NEW')\n            return 'open';\n        if (status === 'PARTIALLY_FILLED')\n            return 'open';\n        if (status === 'FILLED')\n            return 'closed';\n        if (status === 'CANCELED')\n            return 'canceled';\n        return status.toLowerCase ();\n    }\n\n    parseOrder (order, market = undefined) {\n        let status = this.parseOrderStatus (order['status']);\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else {\n            let id = order['symbol'];\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            }\n        }\n        let timestamp = undefined;\n        if ('time' in order)\n            timestamp = order['time'];\n        else if ('transactTime' in order)\n            timestamp = order['transactTime'];\n        else\n            throw new ExchangeError (this.id + ' malformed order: ' + this.json (order));\n        let price = parseFloat (order['price']);\n        let amount = parseFloat (order['origQty']);\n        let filled = this.safeFloat (order, 'executedQty', 0.0);\n        let remaining = Math.max (amount - filled, 0.0);\n        let result = {\n            'info': order,\n            'id': order['orderId'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': order['type'].toLowerCase (),\n            'side': order['side'].toLowerCase (),\n            'price': price,\n            'amount': amount,\n            'cost': price * amount,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': undefined,\n        };\n        return result;\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let order = {\n            'symbol': market['id'],\n            'quantity': this.amountToString (symbol, amount),\n            'type': type.toUpperCase (),\n            'side': side.toUpperCase (),\n        };\n        if (type === 'limit') {\n            order = this.extend (order, {\n                'price': this.priceToPrecision (symbol, price),\n                'timeInForce': 'GTC', // 'GTC' = Good To Cancel (default), 'IOC' = Immediate Or Cancel\n            });\n        }\n        let response = await this.privatePostOrder (this.extend (order, params));\n        return this.parseOrder (response);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrder requires a symbol param');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.privateGetOrder (this.extend ({\n            'symbol': market['id'],\n            'orderId': parseInt (id),\n        }, params));\n        return this.parseOrder (response, market);\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrders requires a symbol param');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n        };\n        if (limit)\n            request['limit'] = limit;\n        let response = await this.privateGetAllOrders (this.extend (request, params));\n        return this.parseOrders (response, market, since, limit);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOpenOrders requires a symbol param');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.privateGetOpenOrders (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseOrders (response, market, since, limit);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, since, limit, params);\n        return this.filterBy (orders, 'status', 'closed');\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' cancelOrder requires a symbol argument');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = undefined;\n        try {\n            response = await this.privateDeleteOrder (this.extend ({\n                'symbol': market['id'],\n                'orderId': parseInt (id),\n                // 'origClientOrderId': id,\n            }, params));\n        } catch (e) {\n            if (this.last_http_response.indexOf ('UNKNOWN_ORDER') >= 0)\n                throw new OrderNotFound (this.id + ' cancelOrder() error: ' + this.last_http_response);\n            throw e;\n        }\n        return response;\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchMyTrades requires a symbol argument');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n        };\n        if (limit)\n            request['limit'] = limit;\n        let response = await this.privateGetMyTrades (this.extend (request, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    commonCurrencyCode (currency) {\n        if (currency === 'BCC')\n            return 'BCH';\n        return currency;\n    }\n\n    currencyId (currency) {\n        if (currency === 'BCH')\n            return 'BCC';\n        return currency;\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        let response = await this.wapiGetDepositAddress (this.extend ({\n            'asset': this.currencyId (currency),\n        }, params));\n        if ('success' in response) {\n            if (response['success']) {\n                let address = this.safeString (response, 'address');\n                return {\n                    'currency': currency,\n                    'address': address,\n                    'status': 'ok',\n                    'info': response,\n                };\n            }\n        }\n        throw new ExchangeError (this.id + ' fetchDepositAddress failed: ' + this.last_http_response);\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        let request = {\n            'asset': this.currencyId (currency),\n            'address': address,\n            'amount': parseFloat (amount),\n            'name': address,\n        };\n        if (tag)\n            request['addressTag'] = tag;\n        let response = await this.wapiPostWithdraw (this.extend (request, params));\n        return {\n            'info': response,\n            'id': this.safeString (response, 'id'),\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        url += '/' + path;\n        if (api === 'wapi')\n            url += '.html';\n        // v1 special case for userDataStream\n        if (path === 'userDataStream') {\n            body = this.urlencode (params);\n            headers = {\n                'X-MBX-APIKEY': this.apiKey,\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        } else if ((api === 'private') || (api === 'wapi')) {\n            this.checkRequiredCredentials ();\n            let nonce = this.milliseconds ();\n            let query = this.urlencode (this.extend ({\n                'timestamp': nonce,\n                'recvWindow': 100000,\n            }, params));\n            let signature = this.hmac (this.encode (query), this.encode (this.secret));\n            query += '&' + 'signature=' + signature;\n            headers = {\n                'X-MBX-APIKEY': this.apiKey,\n            };\n            if ((method === 'GET') || (api === 'wapi')) {\n                url += '?' + query;\n            } else {\n                body = query;\n                headers['Content-Type'] = 'application/x-www-form-urlencoded';\n            }\n        } else {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code >= 400) {\n            if (code === 418)\n                throw new DDoSProtection (this.id + ' ' + code.toString () + ' ' + reason + ' ' + body);\n            if (body.indexOf ('Price * QTY is zero or less') >= 0)\n                throw new InvalidOrder (this.id + ' order cost = amount * price is zero or less ' + body);\n            if (body.indexOf ('MIN_NOTIONAL') >= 0)\n                throw new InvalidOrder (this.id + ' order cost = amount * price is too small ' + body);\n            if (body.indexOf ('LOT_SIZE') >= 0)\n                throw new InvalidOrder (this.id + ' order amount should be evenly divisible by lot size, use this.amountToLots (symbol, amount) ' + body);\n            if (body.indexOf ('PRICE_FILTER') >= 0)\n                throw new InvalidOrder (this.id + ' order price exceeds allowed price precision or invalid, use this.priceToPrecision (symbol, amount) ' + body);\n            if (body.indexOf ('Order does not exist') >= 0)\n                throw new OrderNotFound (this.id + ' ' + body);\n        }\n        if (body[0] === '{') {\n            let response = JSON.parse (body);\n            let error = this.safeValue (response, 'code');\n            if (typeof error !== 'undefined') {\n                if (error === -2010) {\n                    throw new InsufficientFunds (this.id + ' ' + this.json (response));\n                } else if (error === -2011) {\n                    throw new OrderNotFound (this.id + ' ' + this.json (response));\n                } else if (error === -1013) { // Invalid quantity\n                    throw new InvalidOrder (this.id + ' ' + this.json (response));\n                }\n            }\n        }\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange')\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bit2c extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bit2c',\n            'name': 'Bit2C',\n            'countries': 'IL', // Israel\n            'rateLimit': 3000,\n            'has': {\n                'CORS': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766119-3593220e-5ece-11e7-8b3a-5a041f6bcc3f.jpg',\n                'api': 'https://www.bit2c.co.il',\n                'www': 'https://www.bit2c.co.il',\n                'doc': [\n                    'https://www.bit2c.co.il/home/api',\n                    'https://github.com/OferE/bit2c',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'Exchanges/{pair}/Ticker',\n                        'Exchanges/{pair}/orderbook',\n                        'Exchanges/{pair}/trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'Account/Balance',\n                        'Account/Balance/v2',\n                        'Merchant/CreateCheckout',\n                        'Order/AccountHistory',\n                        'Order/AddCoinFundsRequest',\n                        'Order/AddFund',\n                        'Order/AddOrder',\n                        'Order/AddOrderMarketPriceBuy',\n                        'Order/AddOrderMarketPriceSell',\n                        'Order/CancelOrder',\n                        'Order/MyOrders',\n                        'Payment/GetMyId',\n                        'Payment/Send',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/NIS': { 'id': 'BtcNis', 'symbol': 'BTC/NIS', 'base': 'BTC', 'quote': 'NIS' },\n                'BCH/NIS': { 'id': 'BchNis', 'symbol': 'BCH/NIS', 'base': 'BCH', 'quote': 'NIS' },\n                'LTC/NIS': { 'id': 'LtcNis', 'symbol': 'LTC/NIS', 'base': 'LTC', 'quote': 'NIS' },\n                'BTG/NIS': { 'id': 'BtgNis', 'symbol': 'BTG/NIS', 'base': 'BTG', 'quote': 'NIS' },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.5 / 100,\n                    'taker': 0.5 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let balance = await this.privatePostAccountBalanceV2 ();\n        let result = { 'info': balance };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            if (currency in balance) {\n                let available = 'AVAILABLE_' + currency;\n                account['free'] = balance[available];\n                account['total'] = balance[currency];\n                account['used'] = account['total'] - account['free'];\n            }\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetExchangesPairOrderbook (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetExchangesPairTicker (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        let timestamp = this.milliseconds ();\n        let averagePrice = parseFloat (ticker['av']);\n        let baseVolume = parseFloat (ticker['a']);\n        let quoteVolume = baseVolume * averagePrice;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['h']),\n            'ask': parseFloat (ticker['l']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['ll']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': averagePrice,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'id': trade['tid'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'order': undefined,\n            'type': undefined,\n            'side': undefined,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetExchangesPairTrades (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let method = 'privatePostOrderAddOrder';\n        let order = {\n            'Amount': amount,\n            'Pair': this.marketId (symbol),\n        };\n        if (type === 'market') {\n            method += 'MarketPrice' + this.capitalize (side);\n        } else {\n            order['Price'] = price;\n            order['Total'] = amount * price;\n            order['IsBid'] = (side === 'buy');\n        }\n        let result = await this[method] (this.extend (order, params));\n        return {\n            'info': result,\n            'id': result['NewOrder']['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostOrderCancelOrder ({ 'id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        if (api === 'public') {\n            url += '.json';\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let query = this.extend ({ 'nonce': nonce }, params);\n            body = this.urlencode (query);\n            let signature = this.hmac (this.encode (body), this.encode (this.secret), 'sha512', 'base64');\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'key': this.apiKey,\n                'sign': this.decode (signature),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { InvalidNonce, InsufficientFunds, AuthenticationError, InvalidOrder, ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitbay extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitbay',\n            'name': 'BitBay',\n            'countries': [ 'PL', 'EU' ], // Poland\n            'rateLimit': 1000,\n            'has': {\n                'CORS': true,\n                'withdraw': true\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766132-978a7bd8-5ece-11e7-9540-bc96d1e9bbb8.jpg',\n                'www': 'https://bitbay.net',\n                'api': {\n                    'public': 'https://bitbay.net/API/Public',\n                    'private': 'https://bitbay.net/API/Trading/tradingApi.php',\n                },\n                'doc': [\n                    'https://bitbay.net/public-api',\n                    'https://bitbay.net/account/tab-api',\n                    'https://github.com/BitBayNet/API',\n                ],\n                'fees': 'https://bitbay.net/en/fees',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        '{id}/all',\n                        '{id}/market',\n                        '{id}/orderbook',\n                        '{id}/ticker',\n                        '{id}/trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'info',\n                        'trade',\n                        'cancel',\n                        'orderbook',\n                        'orders',\n                        'transfer',\n                        'withdraw',\n                        'history',\n                        'transactions',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/USD': { 'id': 'BTCUSD', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD', 'baseId': 'BTC', 'quoteId': 'USD' },\n                'BTC/EUR': { 'id': 'BTCEUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR', 'baseId': 'BTC', 'quoteId': 'EUR' },\n                'BTC/PLN': { 'id': 'BTCPLN', 'symbol': 'BTC/PLN', 'base': 'BTC', 'quote': 'PLN', 'baseId': 'BTC', 'quoteId': 'PLN' },\n                'LTC/USD': { 'id': 'LTCUSD', 'symbol': 'LTC/USD', 'base': 'LTC', 'quote': 'USD', 'baseId': 'LTC', 'quoteId': 'USD' },\n                'LTC/EUR': { 'id': 'LTCEUR', 'symbol': 'LTC/EUR', 'base': 'LTC', 'quote': 'EUR', 'baseId': 'LTC', 'quoteId': 'EUR' },\n                'LTC/PLN': { 'id': 'LTCPLN', 'symbol': 'LTC/PLN', 'base': 'LTC', 'quote': 'PLN', 'baseId': 'LTC', 'quoteId': 'PLN' },\n                'LTC/BTC': { 'id': 'LTCBTC', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC', 'baseId': 'LTC', 'quoteId': 'BTC' },\n                'ETH/USD': { 'id': 'ETHUSD', 'symbol': 'ETH/USD', 'base': 'ETH', 'quote': 'USD', 'baseId': 'ETH', 'quoteId': 'USD' },\n                'ETH/EUR': { 'id': 'ETHEUR', 'symbol': 'ETH/EUR', 'base': 'ETH', 'quote': 'EUR', 'baseId': 'ETH', 'quoteId': 'EUR' },\n                'ETH/PLN': { 'id': 'ETHPLN', 'symbol': 'ETH/PLN', 'base': 'ETH', 'quote': 'PLN', 'baseId': 'ETH', 'quoteId': 'PLN' },\n                'ETH/BTC': { 'id': 'ETHBTC', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC', 'baseId': 'ETH', 'quoteId': 'BTC' },\n                'LSK/USD': { 'id': 'LSKUSD', 'symbol': 'LSK/USD', 'base': 'LSK', 'quote': 'USD', 'baseId': 'LSK', 'quoteId': 'USD' },\n                'LSK/EUR': { 'id': 'LSKEUR', 'symbol': 'LSK/EUR', 'base': 'LSK', 'quote': 'EUR', 'baseId': 'LSK', 'quoteId': 'EUR' },\n                'LSK/PLN': { 'id': 'LSKPLN', 'symbol': 'LSK/PLN', 'base': 'LSK', 'quote': 'PLN', 'baseId': 'LSK', 'quoteId': 'PLN' },\n                'LSK/BTC': { 'id': 'LSKBTC', 'symbol': 'LSK/BTC', 'base': 'LSK', 'quote': 'BTC', 'baseId': 'LSK', 'quoteId': 'BTC' },\n                'BCH/USD': { 'id': 'BCCUSD', 'symbol': 'BCH/USD', 'base': 'BCH', 'quote': 'USD', 'baseId': 'BCC', 'quoteId': 'USD' },\n                'BCH/EUR': { 'id': 'BCCEUR', 'symbol': 'BCH/EUR', 'base': 'BCH', 'quote': 'EUR', 'baseId': 'BCC', 'quoteId': 'EUR' },\n                'BCH/PLN': { 'id': 'BCCPLN', 'symbol': 'BCH/PLN', 'base': 'BCH', 'quote': 'PLN', 'baseId': 'BCC', 'quoteId': 'PLN' },\n                'BCH/BTC': { 'id': 'BCCBTC', 'symbol': 'BCH/BTC', 'base': 'BCH', 'quote': 'BTC', 'baseId': 'BCC', 'quoteId': 'BTC' },\n                'BTG/USD': { 'id': 'BTGUSD', 'symbol': 'BTG/USD', 'base': 'BTG', 'quote': 'USD', 'baseId': 'BTG', 'quoteId': 'USD' },\n                'BTG/EUR': { 'id': 'BTGEUR', 'symbol': 'BTG/EUR', 'base': 'BTG', 'quote': 'EUR', 'baseId': 'BTG', 'quoteId': 'EUR' },\n                'BTG/PLN': { 'id': 'BTGPLN', 'symbol': 'BTG/PLN', 'base': 'BTG', 'quote': 'PLN', 'baseId': 'BTG', 'quoteId': 'PLN' },\n                'BTG/BTC': { 'id': 'BTGBTC', 'symbol': 'BTG/BTC', 'base': 'BTG', 'quote': 'BTC', 'baseId': 'BTG', 'quoteId': 'BTC' },\n                'DASH/USD': { 'id': 'DASHUSD', 'symbol': 'DASH/USD', 'base': 'DASH', 'quote': 'USD', 'baseId': 'DASH', 'quoteId': 'USD' },\n                'DASH/EUR': { 'id': 'DASHEUR', 'symbol': 'DASH/EUR', 'base': 'DASH', 'quote': 'EUR', 'baseId': 'DASH', 'quoteId': 'EUR' },\n                'DASH/PLN': { 'id': 'DASHPLN', 'symbol': 'DASH/PLN', 'base': 'DASH', 'quote': 'PLN', 'baseId': 'DASH', 'quoteId': 'PLN' },\n                'DASH/BTC': { 'id': 'DASHBTC', 'symbol': 'DASH/BTC', 'base': 'DASH', 'quote': 'BTC', 'baseId': 'DASH', 'quoteId': 'BTC' },\n                'GAME/USD': { 'id': 'GAMEUSD', 'symbol': 'GAME/USD', 'base': 'GAME', 'quote': 'USD', 'baseId': 'GAME', 'quoteId': 'USD' },\n                'GAME/EUR': { 'id': 'GAMEEUR', 'symbol': 'GAME/EUR', 'base': 'GAME', 'quote': 'EUR', 'baseId': 'GAME', 'quoteId': 'EUR' },\n                'GAME/PLN': { 'id': 'GAMEPLN', 'symbol': 'GAME/PLN', 'base': 'GAME', 'quote': 'PLN', 'baseId': 'GAME', 'quoteId': 'PLN' },\n                'GAME/BTC': { 'id': 'GAMEBTC', 'symbol': 'GAME/BTC', 'base': 'GAME', 'quote': 'BTC', 'baseId': 'GAME', 'quoteId': 'BTC' },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.3 / 100,\n                    'taker': 0.0043,\n                },\n                'funding': {\n                    'withdraw': {\n                        'BTC': 0.0009,\n                        'LTC': 0.005,\n                        'ETH': 0.00126,\n                        'LSK': 0.2,\n                        'BCH': 0.0006,\n                        'GAME': 0.005,\n                        'DASH': 0.001,\n                        'BTG': 0.0008,\n                        'PLN': 4,\n                        'EUR': 1.5,\n                    },\n                },\n            },\n            'exceptions': {\n                '400': ExchangeError, // At least one parameter wasn't set\n                '401': InvalidOrder, // Invalid order type\n                '402': InvalidOrder, // No orders with specified currencies\n                '403': InvalidOrder, // Invalid payment currency name\n                '404': InvalidOrder, // Error. Wrong transaction type\n                '405': InvalidOrder, // Order with this id doesn't exist\n                '406': InsufficientFunds, // No enough money or crypto\n                // code 407 not specified are not specified in their docs\n                '408': InvalidOrder, // Invalid currency name\n                '501': AuthenticationError, // Invalid public key\n                '502': AuthenticationError, // Invalid sign\n                '503': InvalidNonce, // Invalid moment parameter. Request time doesn't match current server time\n                '504': ExchangeError, // Invalid method\n                '505': AuthenticationError, // Key has no permission for this action\n                '506': AuthenticationError, // Account locked. Please contact with customer service\n                // codes 507 and 508 are not specified in their docs\n                '509': ExchangeError, // The BIC/SWIFT is required for this currency\n                '510': ExchangeError, // Invalid market name\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostInfo ();\n        if ('balances' in response) {\n            let balance = response['balances'];\n            let result = { 'info': balance };\n            let codes = Object.keys (this.currencies);\n            for (let i = 0; i < codes.length; i++) {\n                let code = codes[i];\n                let currency = this.currencies[code];\n                let id = currency['id'];\n                let account = this.account ();\n                if (id in balance) {\n                    account['free'] = parseFloat (balance[id]['available']);\n                    account['used'] = parseFloat (balance[id]['locked']);\n                    account['total'] = this.sum (account['free'], account['used']);\n                }\n                result[code] = account;\n            }\n            return this.parseBalance (result);\n        }\n        throw new ExchangeError (this.id + ' empty balance response ' + this.json (response));\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetIdOrderbook (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetIdTicker (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        let timestamp = this.milliseconds ();\n        let baseVolume = this.safeFloat (ticker, 'volume');\n        let vwap = this.safeFloat (ticker, 'vwap');\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'max'),\n            'low': this.safeFloat (ticker, 'min'),\n            'bid': this.safeFloat (ticker, 'bid'),\n            'ask': this.safeFloat (ticker, 'ask'),\n            'vwap': vwap,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': this.safeFloat (ticker, 'average'),\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['date'] * 1000;\n        return {\n            'id': trade['tid'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['type'],\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetIdTrades (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let market = this.market (symbol);\n        return this.privatePostTrade (this.extend ({\n            'type': side,\n            'currency': market['baseId'],\n            'amount': amount,\n            'payment_currency': market['quoteId'],\n            'rate': price,\n        }, params));\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancel ({ 'id': id });\n    }\n\n    isFiat (currency) {\n        let fiatCurrencies = {\n            'USD': true,\n            'EUR': true,\n            'PLN': true,\n        };\n        if (currency in fiatCurrencies)\n            return true;\n        return false;\n    }\n\n    async withdraw (code, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let method = undefined;\n        let currency = this.currency (code);\n        let request = {\n            'currency': currency['id'],\n            'quantity': amount,\n        };\n        if (this.isFiat (code)) {\n            method = 'privatePostWithdraw';\n            // request['account'] = params['account']; // they demand an account number\n            // request['express'] = params['express']; // whatever it means, they don't explain\n            // request['bic'] = '';\n        } else {\n            method = 'privatePostTransfer';\n            request['address'] = address;\n        }\n        let response = await this[method] (this.extend (request, params));\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        if (api === 'public') {\n            url += '/' + this.implodeParams (path, params) + '.json';\n        } else {\n            this.checkRequiredCredentials ();\n            body = this.urlencode (this.extend ({\n                'method': path,\n                'moment': this.nonce (),\n            }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'API-Key': this.apiKey,\n                'API-Hash': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (httpCode, reason, url, method, headers, body) {\n        if ((typeof body !== 'string') || (body.length < 2))\n            return; // fallback to default error handler\n        if ((body[0] === '{') || (body[0] === '[')) {\n            let response = JSON.parse (body);\n            if ('code' in response) {\n                //\n                // bitbay returns the integer 'success': 1 key from their private API\n                // or an integer 'code' value from 0 to 510 and an error message\n                //\n                //      { 'success': 1, ... }\n                //      { 'code': 502, 'message': 'Invalid sign' }\n                //      { 'code': 0, 'message': 'offer funds not exceeding minimums' }\n                //\n                let code = response['code']; // always an integer\n                const feedback = this.id + ' ' + this.json (response);\n                const exceptions = this.exceptions;\n                if (code in this.exceptions) {\n                    throw new exceptions[code] (feedback);\n                } else {\n                    throw new ExchangeError (feedback);\n                }\n            }\n        }\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitcoincoid extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitcoincoid',\n            'name': 'Bitcoin.co.id',\n            'countries': 'ID', // Indonesia\n            'has': {\n                'CORS': false,\n                'fetchTickers': false,\n                'fetchOHLCV': false,\n                'fetchOrder': true,\n                'fetchOrders': false,\n                'fetchClosedOrders': true,\n                'fetchOpenOrders': true,\n                'fetchMyTrades': false,\n                'fetchCurrencies': false,\n                'withdraw': false,\n            },\n            'version': '1.7', // as of 6 November 2017\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766138-043c7786-5ecf-11e7-882b-809c14f38b53.jpg',\n                'api': {\n                    'public': 'https://vip.bitcoin.co.id/api',\n                    'private': 'https://vip.bitcoin.co.id/tapi',\n                },\n                'www': 'https://www.bitcoin.co.id',\n                'doc': [\n                    'https://vip.bitcoin.co.id/downloads/BITCOINCOID-API-DOCUMENTATION.pdf',\n                    'https://vip.bitcoin.co.id/trade_api',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        '{pair}/ticker',\n                        '{pair}/trades',\n                        '{pair}/depth',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'getInfo',\n                        'transHistory',\n                        'trade',\n                        'tradeHistory',\n                        'getOrder',\n                        'openOrders',\n                        'cancelOrder',\n                        'orderHistory',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/IDR': { 'id': 'btc_idr', 'symbol': 'BTC/IDR', 'base': 'BTC', 'quote': 'IDR', 'baseId': 'btc', 'quoteId': 'idr' },\n                'BCH/IDR': { 'id': 'bch_idr', 'symbol': 'BCH/IDR', 'base': 'BCH', 'quote': 'IDR', 'baseId': 'bch', 'quoteId': 'idr' },\n                'BTG/IDR': { 'id': 'btg_idr', 'symbol': 'BTG/IDR', 'base': 'BTG', 'quote': 'IDR', 'baseId': 'btg', 'quoteId': 'idr' },\n                'ETH/IDR': { 'id': 'eth_idr', 'symbol': 'ETH/IDR', 'base': 'ETH', 'quote': 'IDR', 'baseId': 'eth', 'quoteId': 'idr' },\n                'ETC/IDR': { 'id': 'etc_idr', 'symbol': 'ETC/IDR', 'base': 'ETC', 'quote': 'IDR', 'baseId': 'etc', 'quoteId': 'idr' },\n                'IGNIS/IDR': { 'id': 'ignis_idr', 'symbol': 'IGNIS/IDR', 'base': 'IGNIS', 'quote': 'IDR', 'baseId': 'ignis', 'quoteId': 'idr' },\n                'LTC/IDR': { 'id': 'ltc_idr', 'symbol': 'LTC/IDR', 'base': 'LTC', 'quote': 'IDR', 'baseId': 'ltc', 'quoteId': 'idr' },\n                'NXT/IDR': { 'id': 'nxt_idr', 'symbol': 'NXT/IDR', 'base': 'NXT', 'quote': 'IDR', 'baseId': 'nxt', 'quoteId': 'idr' },\n                'WAVES/IDR': { 'id': 'waves_idr', 'symbol': 'WAVES/IDR', 'base': 'WAVES', 'quote': 'IDR', 'baseId': 'waves', 'quoteId': 'idr' },\n                'XRP/IDR': { 'id': 'xrp_idr', 'symbol': 'XRP/IDR', 'base': 'XRP', 'quote': 'IDR', 'baseId': 'xrp', 'quoteId': 'idr' },\n                'XZC/IDR': { 'id': 'xzc_idr', 'symbol': 'XZC/IDR', 'base': 'XZC', 'quote': 'IDR', 'baseId': 'xzc', 'quoteId': 'idr' },\n                'XLM/IDR': { 'id': 'str_idr', 'symbol': 'XLM/IDR', 'base': 'XLM', 'quote': 'IDR', 'baseId': 'str', 'quoteId': 'idr' },\n                'BTS/BTC': { 'id': 'bts_btc', 'symbol': 'BTS/BTC', 'base': 'BTS', 'quote': 'BTC', 'baseId': 'bts', 'quoteId': 'btc' },\n                'DASH/BTC': { 'id': 'drk_btc', 'symbol': 'DASH/BTC', 'base': 'DASH', 'quote': 'BTC', 'baseId': 'drk', 'quoteId': 'btc' },\n                'DOGE/BTC': { 'id': 'doge_btc', 'symbol': 'DOGE/BTC', 'base': 'DOGE', 'quote': 'BTC', 'baseId': 'doge', 'quoteId': 'btc' },\n                'ETH/BTC': { 'id': 'eth_btc', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC', 'baseId': 'eth', 'quoteId': 'btc' },\n                'LTC/BTC': { 'id': 'ltc_btc', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC', 'baseId': 'ltc', 'quoteId': 'btc' },\n                'NXT/BTC': { 'id': 'nxt_btc', 'symbol': 'NXT/BTC', 'base': 'NXT', 'quote': 'BTC', 'baseId': 'nxt', 'quoteId': 'btc' },\n                'XLM/BTC': { 'id': 'str_btc', 'symbol': 'XLM/BTC', 'base': 'XLM', 'quote': 'BTC', 'baseId': 'str', 'quoteId': 'btc' },\n                'XEM/BTC': { 'id': 'nem_btc', 'symbol': 'XEM/BTC', 'base': 'XEM', 'quote': 'BTC', 'baseId': 'nem', 'quoteId': 'btc' },\n                'XRP/BTC': { 'id': 'xrp_btc', 'symbol': 'XRP/BTC', 'base': 'XRP', 'quote': 'BTC', 'baseId': 'xrp', 'quoteId': 'btc' },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetInfo ();\n        let balance = response['return'];\n        let result = { 'info': balance };\n        let codes = Object.keys (this.currencies);\n        for (let i = 0; i < codes.length; i++) {\n            let code = codes[i];\n            let currency = this.currencies[code];\n            let lowercase = currency['id'];\n            let account = this.account ();\n            account['free'] = this.safeFloat (balance['balance'], lowercase, 0.0);\n            account['used'] = this.safeFloat (balance['balance_hold'], lowercase, 0.0);\n            account['total'] = this.sum (account['free'], account['used']);\n            result[code] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetPairDepth (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'buy', 'sell');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetPairTicker (this.extend ({\n            'pair': market['id'],\n        }, params));\n        let ticker = response['ticker'];\n        let timestamp = parseFloat (ticker['server_time']) * 1000;\n        let baseVolume = 'vol_' + market['baseId'].toLowerCase ();\n        let quoteVolume = 'vol_' + market['quoteId'].toLowerCase ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy']),\n            'ask': parseFloat (ticker['sell']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker[baseVolume]),\n            'quoteVolume': parseFloat (ticker[quoteVolume]),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        return {\n            'id': trade['tid'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['type'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetPairTrades (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    parseOrder (order, market = undefined) {\n        let side = undefined;\n        if ('type' in order)\n            side = order['type'];\n        let status = this.safeString (order, 'status', 'open');\n        if (status == 'filled') {\n            status = 'closed';\n        } else if (status == 'calcelled') {\n            status = 'canceled';\n        }\n        let symbol = undefined;\n        let cost = undefined;\n        let price = this.safeFloat (order, 'price');\n        let amount = undefined;\n        let remaining = undefined;\n        let filled = undefined;\n        if (market) {\n            symbol = market['symbol'];\n            let quoteId = market['quoteId'];\n            let baseId = market['baseId'];\n            if ((market['quoteId'] == 'idr') && ('order_rp' in order))\n                quoteId = 'rp';\n            if ((market['baseId'] == 'idr') && ('remain_rp' in order))\n                baseId = 'rp';\n            cost = this.safeFloat (order, 'order_' + quoteId);\n            if (cost) {\n                amount = cost / price;\n                let remainingCost = this.safeFloat (order, 'remain_' + quoteId);\n                if (typeof remainingCost !== 'undefined') {\n                    remaining = remainingCost / price;\n                    filled = amount - remaining;\n                }\n            } else {\n                amount = this.safeFloat (order, 'order_' + baseId);\n                cost = price * amount;\n                remaining = this.safeFloat (order, 'remain_' + baseId);\n                filled = amount - remaining;\n            }\n        }\n        let average = undefined;\n        if (filled)\n            average = cost / filled;\n        let timestamp = parseInt (order['submit_time']);\n        let fee = undefined;\n        let result = {\n            'info': order,\n            'id': order['order_id'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': 'limit',\n            'side': side,\n            'price': price,\n            'cost': cost,\n            'average': average,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': fee,\n        };\n        return result;\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrder requires a symbol');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.privatePostGetOrder (this.extend ({\n            'pair': market['id'],\n            'order_id': id,\n        }, params));\n        let orders = response['return'];\n        let order = this.parseOrder (this.extend ({ 'id': id }, orders['order']), market);\n        return this.extend ({ 'info': response }, order);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOpenOrders requires a symbol');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'pair': market['id'],\n        };\n        let response = await this.privatePostOpenOrders (this.extend (request, params));\n        let orders = this.parseOrders (response['return']['orders'], market, since, limit);\n        return this.filterOrdersBySymbol (orders, symbol);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrders requires a symbol');\n        await this.loadMarkets ();\n        let request = {};\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['pair'] = market['id'];\n        }\n        let response = await this.privatePostOrderHistory (this.extend (request, params));\n        let orders = this.parseOrders (response['return']['orders'], market, since, limit);\n        orders = this.filterBy (orders, 'status', 'closed');\n        if (symbol)\n            return this.filterOrdersBySymbol (orders, symbol);\n        return orders;\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let order = {\n            'pair': market['id'],\n            'type': side,\n            'price': price,\n        };\n        let base = market['baseId'];\n        order[base] = amount;\n        let result = await this.privatePostTrade (this.extend (order, params));\n        return {\n            'info': result,\n            'id': result['return']['order_id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelOrder (this.extend ({\n            'order_id': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        if (api == 'public') {\n            url += '/' + this.implodeParams (path, params);\n        } else {\n            this.checkRequiredCredentials ();\n            body = this.urlencode (this.extend ({\n                'method': path,\n                'nonce': this.nonce (),\n            }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Key': this.apiKey,\n                'Sign': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response)\n            throw new ExchangeError (this.id + ' ' + response['error']);\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { DDoSProtection, AuthenticationError, ExchangeError, InsufficientFunds, NotSupported, InvalidOrder, OrderNotFound } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitfinex extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitfinex',\n            'name': 'Bitfinex',\n            'countries': 'VG',\n            'version': 'v1',\n            'rateLimit': 1500,\n            // new metainfo interface\n            'has': {\n                'CORS': false,\n                'fetchOHLCV': true,\n                'fetchTickers': true,\n                'fetchOrder': true,\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': true,\n                'fetchMyTrades': true,\n                'withdraw': true,\n                'deposit': true,\n            },\n            'timeframes': {\n                '1m': '1m',\n                '5m': '5m',\n                '15m': '15m',\n                '30m': '30m',\n                '1h': '1h',\n                '3h': '3h',\n                '6h': '6h',\n                '12h': '12h',\n                '1d': '1D',\n                '1w': '7D',\n                '2w': '14D',\n                '1M': '1M',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766244-e328a50c-5ed2-11e7-947b-041416579bb3.jpg',\n                'api': 'https://api.bitfinex.com',\n                'www': 'https://www.bitfinex.com',\n                'doc': [\n                    'https://bitfinex.readme.io/v1/docs',\n                    'https://github.com/bitfinexcom/bitfinex-api-node',\n                ],\n            },\n            'api': {\n                'v2': {\n                    'get': [\n                        'candles/trade:{timeframe}:{symbol}/{section}',\n                        'candles/trade:{timeframe}:{symbol}/last',\n                        'candles/trade:{timeframe}:{symbol}/hist',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'book/{symbol}',\n                        // 'candles/{symbol}',\n                        'lendbook/{currency}',\n                        'lends/{currency}',\n                        'pubticker/{symbol}',\n                        'stats/{symbol}',\n                        'symbols',\n                        'symbols_details',\n                        'tickers',\n                        'today',\n                        'trades/{symbol}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'account_fees',\n                        'account_infos',\n                        'balances',\n                        'basket_manage',\n                        'credits',\n                        'deposit/new',\n                        'funding/close',\n                        'history',\n                        'history/movements',\n                        'key_info',\n                        'margin_infos',\n                        'mytrades',\n                        'mytrades_funding',\n                        'offer/cancel',\n                        'offer/new',\n                        'offer/status',\n                        'offers',\n                        'offers/hist',\n                        'order/cancel',\n                        'order/cancel/all',\n                        'order/cancel/multi',\n                        'order/cancel/replace',\n                        'order/new',\n                        'order/new/multi',\n                        'order/status',\n                        'orders',\n                        'orders/hist',\n                        'position/claim',\n                        'positions',\n                        'summary',\n                        'taken_funds',\n                        'total_taken_funds',\n                        'transfer',\n                        'unused_taken_funds',\n                        'withdraw',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': true,\n                    'percentage': true,\n                    'maker': 0.1 / 100,\n                    'taker': 0.2 / 100,\n                    'tiers': {\n                        'taker': [\n                            [0, 0.2 / 100],\n                            [500000, 0.2 / 100],\n                            [1000000, 0.2 / 100],\n                            [2500000, 0.2 / 100],\n                            [5000000, 0.2 / 100],\n                            [7500000, 0.2 / 100],\n                            [10000000, 0.18 / 100],\n                            [15000000, 0.16 / 100],\n                            [20000000, 0.14 / 100],\n                            [25000000, 0.12 / 100],\n                            [30000000, 0.1 / 100],\n                        ],\n                        'maker': [\n                            [0, 0.1 / 100],\n                            [500000, 0.08 / 100],\n                            [1000000, 0.06 / 100],\n                            [2500000, 0.04 / 100],\n                            [5000000, 0.02 / 100],\n                            [7500000, 0],\n                            [10000000, 0],\n                            [15000000, 0],\n                            [20000000, 0],\n                            [25000000, 0],\n                            [30000000, 0],\n                        ],\n                    },\n                },\n                'funding': {\n                    'tierBased': false, // true for tier-based/progressive\n                    'percentage': false, // fixed commission\n                    'deposit': {\n                        'BTC': 0.0005,\n                        'IOTA': 0.5,\n                        'ETH': 0.01,\n                        'BCH': 0.01,\n                        'LTC': 0.1,\n                        'EOS': 0.1,\n                        'XMR': 0.04,\n                        'SAN': 0.1,\n                        'DASH': 0.01,\n                        'ETC': 0.01,\n                        'XPR': 0.02,\n                        'YYW': 0.1,\n                        'NEO': 0,\n                        'ZEC': 0.1,\n                        'BTG': 0,\n                        'OMG': 0.1,\n                        'DATA': 1,\n                        'QASH': 1,\n                        'ETP': 0.01,\n                        'QTUM': 0.01,\n                        'EDO': 0.5,\n                        'AVT': 0.5,\n                        'USDT': 0,\n                    },\n                    'withdraw': {\n                        'BTC': 0.0005,\n                        'IOTA': 0.5,\n                        'ETH': 0.01,\n                        'BCH': 0.01,\n                        'LTC': 0.1,\n                        'EOS': 0.1,\n                        'XMR': 0.04,\n                        'SAN': 0.1,\n                        'DASH': 0.01,\n                        'ETC': 0.01,\n                        'XPR': 0.02,\n                        'YYW': 0.1,\n                        'NEO': 0,\n                        'ZEC': 0.1,\n                        'BTG': 0,\n                        'OMG': 0.1,\n                        'DATA': 1,\n                        'QASH': 1,\n                        'ETP': 0.01,\n                        'QTUM': 0.01,\n                        'EDO': 0.5,\n                        'AVT': 0.5,\n                        'USDT': 5,\n                    },\n                },\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        const currencies = {\n            'DSH': 'DASH', // Bitfinex names Dash as DSH, instead of DASH\n            'QTM': 'QTUM',\n            'BCC': 'CST_BCC',\n            'BCU': 'CST_BCU',\n            'IOT': 'IOTA',\n            'DAT': 'DATA',\n        };\n        return (currency in currencies) ? currencies[currency] : currency;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetSymbolsDetails ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['pair'].toUpperCase ();\n            let baseId = id.slice (0, 3);\n            let quoteId = id.slice (3, 6);\n            let base = this.commonCurrencyCode (baseId);\n            let quote = this.commonCurrencyCode (quoteId);\n            let symbol = base + '/' + quote;\n            let precision = {\n                'price': market['price_precision'],\n                'amount': market['price_precision'],\n            };\n            let limits = {\n                'amount': {\n                    'min': parseFloat (market['minimum_order_size']),\n                    'max': parseFloat (market['maximum_order_size']),\n                },\n                'price': {\n                    'min': Math.pow (10, -precision['price']),\n                    'max': Math.pow (10, precision['price']),\n                },\n            };\n            limits['cost'] = {\n                'min': limits['amount']['min'] * limits['price']['min'],\n                'max': undefined,\n            };\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'baseId': baseId,\n                'quoteId': quoteId,\n                'active': true,\n                'precision': precision,\n                'limits': limits,\n                'lot': Math.pow (10, -precision['amount']),\n                'info': market,\n            }));\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balanceType = this.safeString (params, 'type', 'exchange');\n        let balances = await this.privatePostBalances ();\n        let result = { 'info': balances };\n        for (let i = 0; i < balances.length; i++) {\n            let balance = balances[i];\n            if (balance['type'] === balanceType) {\n                let currency = balance['currency'];\n                let uppercase = currency.toUpperCase ();\n                uppercase = this.commonCurrencyCode (uppercase);\n                let account = this.account ();\n                account['free'] = parseFloat (balance['available']);\n                account['total'] = parseFloat (balance['amount']);\n                account['used'] = account['total'] - account['free'];\n                result[uppercase] = account;\n            }\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetBookSymbol (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'amount');\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTickers (params);\n        let result = {};\n        for (let i = 0; i < tickers.length; i++) {\n            let ticker = tickers[i];\n            if ('pair' in ticker) {\n                let id = ticker['pair'];\n                if (id in this.markets_by_id) {\n                    let market = this.markets_by_id[id];\n                    let symbol = market['symbol'];\n                    result[symbol] = this.parseTicker (ticker, market);\n                } else {\n                    throw new ExchangeError (this.id + ' fetchTickers() failed to recognize symbol ' + id + ' ' + this.json (ticker));\n                }\n            } else {\n                throw new ExchangeError (this.id + ' fetchTickers() response not recognized ' + this.json (tickers));\n            }\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetPubtickerSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = parseFloat (ticker['timestamp']) * 1000;\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else if ('pair' in ticker) {\n            let id = ticker['pair'];\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            } else {\n                throw new ExchangeError (this.id + ' unrecognized ticker symbol ' + id + ' ' + this.json (ticker));\n            }\n        }\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last_price']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': parseFloat (ticker['mid']),\n            'baseVolume': parseFloat (ticker['volume']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (parseFloat (trade['timestamp'])) * 1000;\n        let side = trade['type'].toLowerCase ();\n        let orderId = this.safeString (trade, 'order_id');\n        let price = parseFloat (trade['price']);\n        let amount = parseFloat (trade['amount']);\n        let cost = price * amount;\n        return {\n            'id': trade['tid'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'order': orderId,\n            'side': side,\n            'price': price,\n            'amount': amount,\n            'cost': cost,\n            'fee': undefined,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTradesSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = { 'symbol': market['id'] };\n        if (limit) {\n            request['limit_trades'] = limit;\n        }\n        if (since) {\n            request['timestamp'] = parseInt (since / 1000);\n        }\n        let response = await this.privatePostMytrades (this.extend (request, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let orderType = type;\n        if ((type === 'limit') || (type === 'market'))\n            orderType = 'exchange ' + type;\n        // amount = this.amountToPrecision (symbol, amount);\n        let order = {\n            'symbol': this.marketId (symbol),\n            'amount': amount.toString (),\n            'side': side,\n            'type': orderType,\n            'ocoorder': false,\n            'buy_price_oco': 0,\n            'sell_price_oco': 0,\n        };\n        if (type === 'market') {\n            order['price'] = this.nonce ().toString ();\n        } else {\n            // price = this.priceToPrecision (symbol, price);\n            order['price'] = price.toString ();\n        }\n        let result = await this.privatePostOrderNew (this.extend (order, params));\n        return this.parseOrder (result);\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostOrderCancel ({ 'order_id': parseInt (id) });\n    }\n\n    parseOrder (order, market = undefined) {\n        let side = order['side'];\n        let open = order['is_live'];\n        let canceled = order['is_cancelled'];\n        let status = undefined;\n        if (open) {\n            status = 'open';\n        } else if (canceled) {\n            status = 'canceled';\n        } else {\n            status = 'closed';\n        }\n        let symbol = undefined;\n        if (!market) {\n            let exchange = order['symbol'].toUpperCase ();\n            if (exchange in this.markets_by_id) {\n                market = this.markets_by_id[exchange];\n            }\n        }\n        if (market)\n            symbol = market['symbol'];\n        let orderType = order['type'];\n        let exchange = orderType.indexOf ('exchange ') >= 0;\n        if (exchange) {\n            let parts = order['type'].split (' ');\n            orderType = parts[1];\n        }\n        let timestamp = parseInt (parseFloat (order['timestamp']) * 1000);\n        let result = {\n            'info': order,\n            'id': order['id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': orderType,\n            'side': side,\n            'price': this.safeFloat (order, 'price'),\n            'average': parseFloat (order['avg_execution_price']),\n            'amount': parseFloat (order['original_amount']),\n            'remaining': parseFloat (order['remaining_amount']),\n            'filled': parseFloat (order['executed_amount']),\n            'status': status,\n            'fee': undefined,\n        };\n        return result;\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostOrders (params);\n        let orders = this.parseOrders (response, undefined, since, limit);\n        if (symbol)\n            orders = this.filterBy (orders, 'symbol', symbol);\n        return orders;\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        if (limit)\n            request['limit'] = limit;\n        let response = await this.privatePostOrdersHist (this.extend (request, params));\n        let orders = this.parseOrders (response, undefined, since, limit);\n        if (symbol)\n            orders = this.filterBy (orders, 'symbol', symbol);\n        orders = this.filterBy (orders, 'status', 'closed');\n        return orders;\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostOrderStatus (this.extend ({\n            'order_id': parseInt (id),\n        }, params));\n        return this.parseOrder (response);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv[0],\n            ohlcv[1],\n            ohlcv[3],\n            ohlcv[4],\n            ohlcv[2],\n            ohlcv[5],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let v2id = 't' + market['id'];\n        let request = {\n            'symbol': v2id,\n            'timeframe': this.timeframes[timeframe],\n            'sort': 1,\n        };\n        if (limit)\n            request['limit'] = limit;\n        if (since)\n            request['start'] = since;\n        request = this.extend (request, params);\n        let response = await this.v2GetCandlesTradeTimeframeSymbolHist (request);\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    getCurrencyName (currency) {\n        if (currency === 'BTC') {\n            return 'bitcoin';\n        } else if (currency === 'LTC') {\n            return 'litecoin';\n        } else if (currency === 'ETH') {\n            return 'ethereum';\n        } else if (currency === 'ETC') {\n            return 'ethereumc';\n        } else if (currency === 'OMNI') {\n            return 'mastercoin'; // ???\n        } else if (currency === 'ZEC') {\n            return 'zcash';\n        } else if (currency === 'XMR') {\n            return 'monero';\n        } else if (currency === 'USD') {\n            return 'wire';\n        } else if (currency === 'DASH') {\n            return 'dash';\n        } else if (currency === 'XRP') {\n            return 'ripple';\n        } else if (currency === 'EOS') {\n            return 'eos';\n        } else if (currency === 'BCH') {\n            return 'bcash';\n        } else if (currency === 'USDT') {\n            return 'tetheruso';\n        }\n        throw new NotSupported (this.id + ' ' + currency + ' not supported for withdrawal');\n    }\n\n    async createDepositAddress (currency, params = {}) {\n        let response = await this.fetchDepositAddress (currency, this.extend ({\n            'renew': 1,\n        }, params));\n        return {\n            'currency': currency,\n            'address': response['address'],\n            'status': 'ok',\n            'info': response['info'],\n        };\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        let name = this.getCurrencyName (currency);\n        let request = {\n            'method': name,\n            'wallet_name': 'exchange',\n            'renew': 0, // a value of 1 will generate a new address\n        };\n        let response = await this.privatePostDepositNew (this.extend (request, params));\n        return {\n            'currency': currency,\n            'address': response['address'],\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        let name = this.getCurrencyName (currency);\n        let request = {\n            'withdraw_type': name,\n            'walletselected': 'exchange',\n            'amount': amount.toString (),\n            'address': address,\n        };\n        if (tag)\n            request['payment_id'] = tag;\n        let responses = await this.privatePostWithdraw (this.extend (request, params));\n        let response = responses[0];\n        return {\n            'info': response,\n            'id': response['withdrawal_id'],\n        };\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let request = '/' + this.implodeParams (path, params);\n        if (api === 'v2') {\n            request = '/' + api + request;\n        } else {\n            request = '/' + this.version + request;\n        }\n        let query = this.omit (params, this.extractParams (path));\n        let url = this.urls['api'] + request;\n        if ((api === 'public') || (path.indexOf ('/hist') >= 0)) {\n            if (Object.keys (query).length) {\n                let suffix = '?' + this.urlencode (query);\n                url += suffix;\n                request += suffix;\n            }\n        }\n        if (api === 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            query = this.extend ({\n                'nonce': nonce.toString (),\n                'request': request,\n            }, query);\n            query = this.json (query);\n            query = this.encode (query);\n            let payload = this.stringToBase64 (query);\n            let secret = this.encode (this.secret);\n            let signature = this.hmac (payload, secret, 'sha384');\n            headers = {\n                'X-BFX-APIKEY': this.apiKey,\n                'X-BFX-PAYLOAD': this.decode (payload),\n                'X-BFX-SIGNATURE': signature,\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (body.length < 2)\n            return;\n        if (code >= 400) {\n            if (body[0] === '{') {\n                let response = JSON.parse (body);\n                if ('message' in response) {\n                    let message = response['message'];\n                    let error = this.id + ' ' + message;\n                    if (message.indexOf ('Key price should be a decimal number') >= 0) {\n                        throw new InvalidOrder (error);\n                    } else if (message.indexOf ('Invalid order: not enough exchange balance') >= 0) {\n                        throw new InsufficientFunds (error);\n                    } else if (message === 'Order could not be cancelled.') {\n                        throw new OrderNotFound (error);\n                    } else if (message.indexOf ('Invalid order') >= 0) {\n                        throw new InvalidOrder (error);\n                    } else if (message === 'Order price must be positive.') {\n                        throw new InvalidOrder (error);\n                    } else if (message.indexOf ('Key amount should be a decimal number') >= 0) {\n                        throw new InvalidOrder (error);\n                    } else if (message === 'No such order found.') {\n                        throw new OrderNotFound (error);\n                    } else if (message === 'Could not find a key matching the given X-BFX-APIKEY.') {\n                        throw new AuthenticationError (error);\n                    }\n                } else if ('error' in response) {\n                    let code = response['error'];\n                    let error = this.id + ' ' + code;\n                    if (code === 'ERR_RATE_LIMIT')\n                        throw new DDoSProtection (error);\n                }\n            }\n        }\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('message' in response) {\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        return response;\n    }\n};\n","'use strict';\n\n// ---------------------------------------------------------------------------\n\nconst bitfinex = require ('./bitfinex.js');\nconst { ExchangeError, NotSupported, InsufficientFunds } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class bitfinex2 extends bitfinex {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitfinex2',\n            'name': 'Bitfinex v2',\n            'countries': 'VG',\n            'version': 'v2',\n            // new metainfo interface\n            'has': {\n                'CORS': true,\n                'createOrder': false,\n                'fetchOHLCV': true,\n                'fetchTickers': true,\n                'fetchOrder': true,\n                'fetchOpenOrders': false,\n                'fetchClosedOrders': false,\n                'withdraw': true,\n                'deposit': false,\n            },\n            'timeframes': {\n                '1m': '1m',\n                '5m': '5m',\n                '15m': '15m',\n                '30m': '30m',\n                '1h': '1h',\n                '3h': '3h',\n                '6h': '6h',\n                '12h': '12h',\n                '1d': '1D',\n                '1w': '7D',\n                '2w': '14D',\n                '1M': '1M',\n            },\n            'rateLimit': 1500,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766244-e328a50c-5ed2-11e7-947b-041416579bb3.jpg',\n                'api': 'https://api.bitfinex.com',\n                'www': 'https://www.bitfinex.com',\n                'doc': [\n                    'https://bitfinex.readme.io/v2/docs',\n                    'https://github.com/bitfinexcom/bitfinex-api-node',\n                ],\n                'fees': 'https://www.bitfinex.com/fees',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'platform/status',\n                        'tickers',\n                        'ticker/{symbol}',\n                        'trades/{symbol}/hist',\n                        'book/{symbol}/{precision}',\n                        'book/{symbol}/P0',\n                        'book/{symbol}/P1',\n                        'book/{symbol}/P2',\n                        'book/{symbol}/P3',\n                        'book/{symbol}/R0',\n                        'symbols_details',\n                        'stats1/{key}:{size}:{symbol}/{side}/{section}',\n                        'stats1/{key}:{size}:{symbol}/long/last',\n                        'stats1/{key}:{size}:{symbol}/long/hist',\n                        'stats1/{key}:{size}:{symbol}/short/last',\n                        'stats1/{key}:{size}:{symbol}/short/hist',\n                        'candles/trade:{timeframe}:{symbol}/{section}',\n                        'candles/trade:{timeframe}:{symbol}/last',\n                        'candles/trade:{timeframe}:{symbol}/hist',\n                    ],\n                    'post': [\n                        'calc/trade/avg',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'auth/r/wallets',\n                        'auth/r/orders/{symbol}',\n                        'auth/r/orders/{symbol}/new',\n                        'auth/r/orders/{symbol}/hist',\n                        'auth/r/order/{symbol}:{id}/trades',\n                        'auth/r/trades/{symbol}/hist',\n                        'auth/r/positions',\n                        'auth/r/funding/offers/{symbol}',\n                        'auth/r/funding/offers/{symbol}/hist',\n                        'auth/r/funding/loans/{symbol}',\n                        'auth/r/funding/loans/{symbol}/hist',\n                        'auth/r/funding/credits/{symbol}',\n                        'auth/r/funding/credits/{symbol}/hist',\n                        'auth/r/funding/trades/{symbol}/hist',\n                        'auth/r/info/margin/{key}',\n                        'auth/r/info/funding/{key}',\n                        'auth/r/movements/{currency}/hist',\n                        'auth/r/stats/perf:{timeframe}/hist',\n                        'auth/r/alerts',\n                        'auth/w/alert/set',\n                        'auth/w/alert/{type}:{symbol}:{price}/del',\n                        'auth/calc/order/avail',\n                    ],\n                },\n            },\n            'markets': {\n                'AVT/BTC': { 'id': 'tAVTBTC', 'symbol': 'AVT/BTC', 'base': 'AVT', 'quote': 'BTC', 'baseId': 'tAVT', 'quoteId': 'tBTC' },\n                'AVT/ETH': { 'id': 'tAVTETH', 'symbol': 'AVT/ETH', 'base': 'AVT', 'quote': 'ETH', 'baseId': 'tAVT', 'quoteId': 'tETH' },\n                'AVT/USD': { 'id': 'tAVTUSD', 'symbol': 'AVT/USD', 'base': 'AVT', 'quote': 'USD', 'baseId': 'tAVT', 'quoteId': 'zUSD' },\n                'CST_BCC/BTC': { 'id': 'tBCCBTC', 'symbol': 'CST_BCC/BTC', 'base': 'CST_BCC', 'quote': 'BTC', 'baseId': 'tBCC', 'quoteId': 'tBTC' },\n                'CST_BCC/USD': { 'id': 'tBCCUSD', 'symbol': 'CST_BCC/USD', 'base': 'CST_BCC', 'quote': 'USD', 'baseId': 'tBCC', 'quoteId': 'zUSD' },\n                'BCH/BTC': { 'id': 'tBCHBTC', 'symbol': 'BCH/BTC', 'base': 'BCH', 'quote': 'BTC', 'baseId': 'tBCH', 'quoteId': 'tBTC' },\n                'BCH/ETH': { 'id': 'tBCHETH', 'symbol': 'BCH/ETH', 'base': 'BCH', 'quote': 'ETH', 'baseId': 'tBCH', 'quoteId': 'tETH' },\n                'BCH/USD': { 'id': 'tBCHUSD', 'symbol': 'BCH/USD', 'base': 'BCH', 'quote': 'USD', 'baseId': 'tBCH', 'quoteId': 'zUSD' },\n                'CST_BCU/BTC': { 'id': 'tBCUBTC', 'symbol': 'CST_BCU/BTC', 'base': 'CST_BCU', 'quote': 'BTC', 'baseId': 'tBCU', 'quoteId': 'tBTC' },\n                'CST_BCU/USD': { 'id': 'tBCUUSD', 'symbol': 'CST_BCU/USD', 'base': 'CST_BCU', 'quote': 'USD', 'baseId': 'tBCU', 'quoteId': 'zUSD' },\n                'BT1/BTC': { 'id': 'tBT1BTC', 'symbol': 'BT1/BTC', 'base': 'BT1', 'quote': 'BTC', 'baseId': 'tBT1', 'quoteId': 'tBTC' },\n                'BT1/USD': { 'id': 'tBT1USD', 'symbol': 'BT1/USD', 'base': 'BT1', 'quote': 'USD', 'baseId': 'tBT1', 'quoteId': 'zUSD' },\n                'BT2/BTC': { 'id': 'tBT2BTC', 'symbol': 'BT2/BTC', 'base': 'BT2', 'quote': 'BTC', 'baseId': 'tBT2', 'quoteId': 'tBTC' },\n                'BT2/USD': { 'id': 'tBT2USD', 'symbol': 'BT2/USD', 'base': 'BT2', 'quote': 'USD', 'baseId': 'tBT2', 'quoteId': 'zUSD' },\n                'BTC/USD': { 'id': 'tBTCUSD', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD', 'baseId': 'tBTC', 'quoteId': 'zUSD' },\n                'BTC/EUR': { 'id': 'tBTCEUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR', 'baseId': 'tBTC', 'quoteId': 'zEUR' },\n                'BTG/BTC': { 'id': 'tBTGBTC', 'symbol': 'BTG/BTC', 'base': 'BTG', 'quote': 'BTC', 'baseId': 'tBTG', 'quoteId': 'tBTC' },\n                'BTG/USD': { 'id': 'tBTGUSD', 'symbol': 'BTG/USD', 'base': 'BTG', 'quote': 'USD', 'baseId': 'tBTG', 'quoteId': 'zUSD' },\n                'DASH/BTC': { 'id': 'tDSHBTC', 'symbol': 'DASH/BTC', 'base': 'DASH', 'quote': 'BTC', 'baseId': 'tDASH', 'quoteId': 'tBTC' },\n                'DASH/USD': { 'id': 'tDSHUSD', 'symbol': 'DASH/USD', 'base': 'DASH', 'quote': 'USD', 'baseId': 'tDASH', 'quoteId': 'zUSD' },\n                'DAT/BTC': { 'id': 'tDATBTC', 'symbol': 'DAT/BTC', 'base': 'DAT', 'quote': 'BTC', 'baseId': 'tDAT', 'quoteId': 'tBTC' },\n                'DAT/ETH': { 'id': 'tDATETH', 'symbol': 'DAT/ETH', 'base': 'DAT', 'quote': 'ETH', 'baseId': 'tDAT', 'quoteId': 'tETH' },\n                'DAT/USD': { 'id': 'tDATUSD', 'symbol': 'DAT/USD', 'base': 'DAT', 'quote': 'USD', 'baseId': 'tDAT', 'quoteId': 'zUSD' },\n                'EDO/BTC': { 'id': 'tEDOBTC', 'symbol': 'EDO/BTC', 'base': 'EDO', 'quote': 'BTC', 'baseId': 'tEDO', 'quoteId': 'tBTC' },\n                'EDO/ETH': { 'id': 'tEDOETH', 'symbol': 'EDO/ETH', 'base': 'EDO', 'quote': 'ETH', 'baseId': 'tEDO', 'quoteId': 'tETH' },\n                'EDO/USD': { 'id': 'tEDOUSD', 'symbol': 'EDO/USD', 'base': 'EDO', 'quote': 'USD', 'baseId': 'tEDO', 'quoteId': 'zUSD' },\n                'EOS/BTC': { 'id': 'tEOSBTC', 'symbol': 'EOS/BTC', 'base': 'EOS', 'quote': 'BTC', 'baseId': 'tEOS', 'quoteId': 'tBTC' },\n                'EOS/ETH': { 'id': 'tEOSETH', 'symbol': 'EOS/ETH', 'base': 'EOS', 'quote': 'ETH', 'baseId': 'tEOS', 'quoteId': 'tETH' },\n                'EOS/USD': { 'id': 'tEOSUSD', 'symbol': 'EOS/USD', 'base': 'EOS', 'quote': 'USD', 'baseId': 'tEOS', 'quoteId': 'zUSD' },\n                'ETC/BTC': { 'id': 'tETCBTC', 'symbol': 'ETC/BTC', 'base': 'ETC', 'quote': 'BTC', 'baseId': 'tETC', 'quoteId': 'tBTC' },\n                'ETC/USD': { 'id': 'tETCUSD', 'symbol': 'ETC/USD', 'base': 'ETC', 'quote': 'USD', 'baseId': 'tETC', 'quoteId': 'zUSD' },\n                'ETH/BTC': { 'id': 'tETHBTC', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC', 'baseId': 'tETH', 'quoteId': 'tBTC' },\n                'ETH/USD': { 'id': 'tETHUSD', 'symbol': 'ETH/USD', 'base': 'ETH', 'quote': 'USD', 'baseId': 'tETH', 'quoteId': 'zUSD' },\n                'ETP/BTC': { 'id': 'tETPBTC', 'symbol': 'ETP/BTC', 'base': 'ETP', 'quote': 'BTC', 'baseId': 'tETP', 'quoteId': 'tBTC' },\n                'ETP/ETH': { 'id': 'tETPETH', 'symbol': 'ETP/ETH', 'base': 'ETP', 'quote': 'ETH', 'baseId': 'tETP', 'quoteId': 'tETH' },\n                'ETP/USD': { 'id': 'tETPUSD', 'symbol': 'ETP/USD', 'base': 'ETP', 'quote': 'USD', 'baseId': 'tETP', 'quoteId': 'zUSD' },\n                'IOTA/BTC': { 'id': 'tIOTBTC', 'symbol': 'IOTA/BTC', 'base': 'IOTA', 'quote': 'BTC', 'baseId': 'tIOTA', 'quoteId': 'tBTC' },\n                'IOTA/ETH': { 'id': 'tIOTETH', 'symbol': 'IOTA/ETH', 'base': 'IOTA', 'quote': 'ETH', 'baseId': 'tIOTA', 'quoteId': 'tETH' },\n                'IOTA/USD': { 'id': 'tIOTUSD', 'symbol': 'IOTA/USD', 'base': 'IOTA', 'quote': 'USD', 'baseId': 'tIOTA', 'quoteId': 'zUSD' },\n                'LTC/BTC': { 'id': 'tLTCBTC', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC', 'baseId': 'tLTC', 'quoteId': 'tBTC' },\n                'LTC/USD': { 'id': 'tLTCUSD', 'symbol': 'LTC/USD', 'base': 'LTC', 'quote': 'USD', 'baseId': 'tLTC', 'quoteId': 'zUSD' },\n                'NEO/BTC': { 'id': 'tNEOBTC', 'symbol': 'NEO/BTC', 'base': 'NEO', 'quote': 'BTC', 'baseId': 'tNEO', 'quoteId': 'tBTC' },\n                'NEO/ETH': { 'id': 'tNEOETH', 'symbol': 'NEO/ETH', 'base': 'NEO', 'quote': 'ETH', 'baseId': 'tNEO', 'quoteId': 'tETH' },\n                'NEO/USD': { 'id': 'tNEOUSD', 'symbol': 'NEO/USD', 'base': 'NEO', 'quote': 'USD', 'baseId': 'tNEO', 'quoteId': 'zUSD' },\n                'OMG/BTC': { 'id': 'tOMGBTC', 'symbol': 'OMG/BTC', 'base': 'OMG', 'quote': 'BTC', 'baseId': 'tOMG', 'quoteId': 'tBTC' },\n                'OMG/ETH': { 'id': 'tOMGETH', 'symbol': 'OMG/ETH', 'base': 'OMG', 'quote': 'ETH', 'baseId': 'tOMG', 'quoteId': 'tETH' },\n                'OMG/USD': { 'id': 'tOMGUSD', 'symbol': 'OMG/USD', 'base': 'OMG', 'quote': 'USD', 'baseId': 'tOMG', 'quoteId': 'zUSD' },\n                'QTUM/BTC': { 'id': 'tQTMBTC', 'symbol': 'QTUM/BTC', 'base': 'QTUM', 'quote': 'BTC', 'baseId': 'tQTUM', 'quoteId': 'tBTC' },\n                'QTUM/ETH': { 'id': 'tQTMETH', 'symbol': 'QTUM/ETH', 'base': 'QTUM', 'quote': 'ETH', 'baseId': 'tQTUM', 'quoteId': 'tETH' },\n                'QTUM/USD': { 'id': 'tQTMUSD', 'symbol': 'QTUM/USD', 'base': 'QTUM', 'quote': 'USD', 'baseId': 'tQTUM', 'quoteId': 'zUSD' },\n                'RRT/BTC': { 'id': 'tRRTBTC', 'symbol': 'RRT/BTC', 'base': 'RRT', 'quote': 'BTC', 'baseId': 'tRRT', 'quoteId': 'tBTC' },\n                'RRT/USD': { 'id': 'tRRTUSD', 'symbol': 'RRT/USD', 'base': 'RRT', 'quote': 'USD', 'baseId': 'tRRT', 'quoteId': 'zUSD' },\n                'SAN/BTC': { 'id': 'tSANBTC', 'symbol': 'SAN/BTC', 'base': 'SAN', 'quote': 'BTC', 'baseId': 'tSAN', 'quoteId': 'tBTC' },\n                'SAN/ETH': { 'id': 'tSANETH', 'symbol': 'SAN/ETH', 'base': 'SAN', 'quote': 'ETH', 'baseId': 'tSAN', 'quoteId': 'tETH' },\n                'SAN/USD': { 'id': 'tSANUSD', 'symbol': 'SAN/USD', 'base': 'SAN', 'quote': 'USD', 'baseId': 'tSAN', 'quoteId': 'zUSD' },\n                'XMR/BTC': { 'id': 'tXMRBTC', 'symbol': 'XMR/BTC', 'base': 'XMR', 'quote': 'BTC', 'baseId': 'tXMR', 'quoteId': 'tBTC' },\n                'XMR/USD': { 'id': 'tXMRUSD', 'symbol': 'XMR/USD', 'base': 'XMR', 'quote': 'USD', 'baseId': 'tXMR', 'quoteId': 'zUSD' },\n                'XRP/BTC': { 'id': 'tXRPBTC', 'symbol': 'XRP/BTC', 'base': 'XRP', 'quote': 'BTC', 'baseId': 'tXRP', 'quoteId': 'tBTC' },\n                'XRP/USD': { 'id': 'tXRPUSD', 'symbol': 'XRP/USD', 'base': 'XRP', 'quote': 'USD', 'baseId': 'tXRP', 'quoteId': 'zUSD' },\n                'ZEC/BTC': { 'id': 'tZECBTC', 'symbol': 'ZEC/BTC', 'base': 'ZEC', 'quote': 'BTC', 'baseId': 'tZEC', 'quoteId': 'tBTC' },\n                'ZEC/USD': { 'id': 'tZECUSD', 'symbol': 'ZEC/USD', 'base': 'ZEC', 'quote': 'USD', 'baseId': 'tZEC', 'quoteId': 'zUSD' },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.1 / 100,\n                    'taker': 0.2 / 100,\n                },\n                'funding': {\n                    'withdraw': {\n                        'BTC': 0.0005,\n                        'BCH': 0.0005,\n                        'ETH': 0.01,\n                        'EOS': 0.1,\n                        'LTC': 0.001,\n                        'OMG': 0.1,\n                        'IOT': 0.0,\n                        'NEO': 0.0,\n                        'ETC': 0.01,\n                        'XRP': 0.02,\n                        'ETP': 0.01,\n                        'ZEC': 0.001,\n                        'BTG': 0.0,\n                        'DASH': 0.01,\n                        'XMR': 0.04,\n                        'QTM': 0.01,\n                        'EDO': 0.5,\n                        'DAT': 1.0,\n                        'AVT': 0.5,\n                        'SAN': 0.1,\n                        'USDT': 5.0,\n                    },\n                },\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        const currencies = {\n            'DSH': 'DASH', // Bitfinex names Dash as DSH, instead of DASH\n            'QTM': 'QTUM',\n            'BCC': 'CST_BCC',\n            'BCU': 'CST_BCU',\n            'IOT': 'IOTA',\n            'DAT': 'DATA',\n        };\n        return (currency in currencies) ? currencies[currency] : currency;\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostAuthRWallets ();\n        let balanceType = this.safeString (params, 'type', 'exchange');\n        let result = { 'info': response };\n        for (let b = 0; b < response.length; b++) {\n            let balance = response[b];\n            let [ accountType, currency, total, interest, available ] = balance;\n            if (accountType === balanceType) {\n                if (currency[0] === 't')\n                    currency = currency.slice (1);\n                let uppercase = currency.toUpperCase ();\n                uppercase = this.commonCurrencyCode (uppercase);\n                let account = this.account ();\n                account['free'] = available;\n                account['total'] = total;\n                if (account['free'])\n                    account['used'] = account['total'] - account['free'];\n                result[uppercase] = account;\n            }\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetBookSymbolPrecision (this.extend ({\n            'symbol': this.marketId (symbol),\n            'precision': 'R0',\n        }, params));\n        let timestamp = this.milliseconds ();\n        let result = {\n            'bids': [],\n            'asks': [],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n        };\n        for (let i = 0; i < orderbook.length; i++) {\n            let order = orderbook[i];\n            let price = order[1];\n            let amount = order[2];\n            let side = (amount > 0) ? 'bids' : 'asks';\n            amount = Math.abs (amount);\n            result[side].push ([ price, amount ]);\n        }\n        result['bids'] = this.sortBy (result['bids'], 0, true);\n        result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let length = ticker.length;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': ticker[length - 2],\n            'low': ticker[length - 1],\n            'bid': ticker[length - 10],\n            'ask': ticker[length - 8],\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': ticker[length - 4],\n            'change': ticker[length - 6],\n            'percentage': ticker[length - 5],\n            'average': undefined,\n            'baseVolume': ticker[length - 3],\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        let tickers = await this.publicGetTickers (this.extend ({\n            'symbols': this.ids.join (','),\n        }, params));\n        let result = {};\n        for (let i = 0; i < tickers.length; i++) {\n            let ticker = tickers[i];\n            let id = ticker[0];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let market = this.markets[symbol];\n        let ticker = await this.publicGetTickerSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let [ id, timestamp, amount, price ] = trade;\n        let side = (amount < 0) ? 'sell' : 'buy';\n        if (amount < 0) {\n            amount = -amount;\n        }\n        return {\n            'id': id.toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': side,\n            'price': price,\n            'amount': amount,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n        };\n        if (since) {\n            request['start'] = since;\n        }\n        if (limit) {\n            request['limit'] = limit;\n        }\n        let response = await this.publicGetTradesSymbolHist (this.extend (request, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n            'timeframe': this.timeframes[timeframe],\n            'sort': 1,\n        };\n        if (limit)\n            request['limit'] = limit;\n        if (since)\n            request['start'] = since;\n        request = this.extend (request, params);\n        let response = await this.publicGetCandlesTradeTimeframeSymbolHist (request);\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        throw new NotSupported (this.id + ' createOrder not implemented yet');\n    }\n\n    cancelOrder (id, symbol = undefined, params = {}) {\n        throw new NotSupported (this.id + ' cancelOrder not implemented yet');\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchOrder not implemented yet');\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        throw new NotSupported (this.id + ' withdraw not implemented yet');\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let request = this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        let url = this.urls['api'] + '/' + request;\n        if (api === 'public') {\n            if (Object.keys (query).length) {\n                url += '?' + this.urlencode (query);\n            }\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            body = this.json (query);\n            let auth = '/api' + '/' + request + nonce + body;\n            let signature = this.hmac (this.encode (auth), this.encode (this.secret), 'sha384');\n            headers = {\n                'bfx-nonce': nonce,\n                'bfx-apikey': this.apiKey,\n                'bfx-signature': signature,\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (response) {\n            if ('message' in response) {\n                if (response['message'].indexOf ('not enough exchange balance') >= 0)\n                    throw new InsufficientFunds (this.id + ' ' + this.json (response));\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n            }\n            return response;\n        } else if (response === '') {\n            throw new ExchangeError (this.id + ' returned empty response');\n        }\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitflyer extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitflyer',\n            'name': 'bitFlyer',\n            'countries': 'JP',\n            'version': 'v1',\n            'rateLimit': 500,\n            'has': {\n                'CORS': false,\n                'withdraw': true\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28051642-56154182-660e-11e7-9b0d-6042d1e6edd8.jpg',\n                'api': 'https://api.bitflyer.jp',\n                'www': 'https://bitflyer.jp',\n                'doc': 'https://bitflyer.jp/API',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'getmarkets',    // or 'markets'\n                        'getboard',      // or 'board'\n                        'getticker',     // or 'ticker'\n                        'getexecutions', // or 'executions'\n                        'gethealth',\n                        'getchats',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'getpermissions',\n                        'getbalance',\n                        'getcollateral',\n                        'getcollateralaccounts',\n                        'getaddresses',\n                        'getcoinins',\n                        'getcoinouts',\n                        'getbankaccounts',\n                        'getdeposits',\n                        'getwithdrawals',\n                        'getchildorders',\n                        'getparentorders',\n                        'getparentorder',\n                        'getexecutions',\n                        'getpositions',\n                        'gettradingcommission',\n                    ],\n                    'post': [\n                        'sendcoin',\n                        'withdraw',\n                        'sendchildorder',\n                        'cancelchildorder',\n                        'sendparentorder',\n                        'cancelparentorder',\n                        'cancelallchildorders',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.25 / 100,\n                    'taker': 0.25 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetMarkets ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['product_code'];\n            let currencies = id.split ('_');\n            let base = undefined;\n            let quote = undefined;\n            let symbol = id;\n            let numCurrencies = currencies.length;\n            if (numCurrencies === 1) {\n                base = symbol.slice (0, 3);\n                quote = symbol.slice (3, 6);\n            } else if (numCurrencies === 2) {\n                base = currencies[0];\n                quote = currencies[1];\n                symbol = base + '/' + quote;\n            } else {\n                base = currencies[1];\n                quote = currencies[2];\n            }\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetBalance ();\n        let balances = {};\n        for (let b = 0; b < response.length; b++) {\n            let account = response[b];\n            let currency = account['currency_code'];\n            balances[currency] = account;\n        }\n        let result = { 'info': response };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            if (currency in balances) {\n                account['total'] = balances[currency]['amount'];\n                account['free'] = balances[currency]['available'];\n                account['used'] = account['total'] - account['free'];\n            }\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetBoard (this.extend ({\n            'product_code': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'size');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let ticker = await this.publicGetTicker (this.extend ({\n            'product_code': this.marketId (symbol),\n        }, params));\n        let timestamp = this.parse8601 (ticker['timestamp']);\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['best_bid']),\n            'ask': parseFloat (ticker['best_ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['ltp']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume_by_product']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let side = undefined;\n        let order = undefined;\n        if ('side' in trade)\n            if (trade['side']) {\n                side = trade['side'].toLowerCase ();\n                let id = side + '_child_order_acceptance_id';\n                if (id in trade)\n                    order = trade[id];\n            }\n        let timestamp = this.parse8601 (trade['exec_date']);\n        return {\n            'id': trade['id'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': order,\n            'type': undefined,\n            'side': side,\n            'price': trade['price'],\n            'amount': trade['size'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetExecutions (this.extend ({\n            'product_code': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'product_code': this.marketId (symbol),\n            'child_order_type': type.toUpperCase (),\n            'side': side.toUpperCase (),\n            'price': price,\n            'size': amount,\n        };\n        let result = await this.privatePostSendchildorder (this.extend (order, params));\n        return {\n            'info': result,\n            'id': result['child_order_acceptance_id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelchildorder (this.extend ({\n            'parent_order_id': id,\n        }, params));\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostWithdraw (this.extend ({\n            'currency_code': currency,\n            'amount': amount,\n            // 'bank_account_id': 1234,\n        }, params));\n        return {\n            'info': response,\n            'id': response['message_id'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let request = '/' + this.version + '/';\n        if (api === 'private')\n            request += 'me/';\n        request += path;\n        if (method === 'GET') {\n            if (Object.keys (params).length)\n                request += '?' + this.urlencode (params);\n        }\n        let url = this.urls['api'] + request;\n        if (api === 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            body = this.json (params);\n            let auth = [ nonce, method, request, body ].join ('');\n            headers = {\n                'ACCESS-KEY': this.apiKey,\n                'ACCESS-TIMESTAMP': nonce,\n                'ACCESS-SIGN': this.hmac (this.encode (auth), this.encode (this.secret)),\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bithumb extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bithumb',\n            'name': 'Bithumb',\n            'countries': 'KR', // South Korea\n            'rateLimit': 500,\n            'has': {\n                'CORS': true,\n                'fetchTickers': true,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30597177-ea800172-9d5e-11e7-804c-b9d4fa9b56b0.jpg',\n                'api': {\n                    'public': 'https://api.bithumb.com/public',\n                    'private': 'https://api.bithumb.com',\n                },\n                'www': 'https://www.bithumb.com',\n                'doc': 'https://www.bithumb.com/u1/US127',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'ticker/{currency}',\n                        'ticker/all',\n                        'orderbook/{currency}',\n                        'orderbook/all',\n                        'recent_transactions/{currency}',\n                        'recent_transactions/all',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'info/account',\n                        'info/balance',\n                        'info/wallet_address',\n                        'info/ticker',\n                        'info/orders',\n                        'info/user_transactions',\n                        'trade/place',\n                        'info/order_detail',\n                        'trade/cancel',\n                        'trade/btc_withdrawal',\n                        'trade/krw_deposit',\n                        'trade/krw_withdrawal',\n                        'trade/market_buy',\n                        'trade/market_sell',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.15 / 100,\n                    'taker': 0.15 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetTickerAll ();\n        let currencies = Object.keys (markets['data']);\n        let result = [];\n        for (let i = 0; i < currencies.length; i++) {\n            let id = currencies[i];\n            if (id !== 'date') {\n                let market = markets['data'][id];\n                let base = id;\n                let quote = 'KRW';\n                let symbol = id + '/' + quote;\n                result.push (this.extend (this.fees['trading'], {\n                    'id': id,\n                    'symbol': symbol,\n                    'base': base,\n                    'quote': quote,\n                    'info': market,\n                    'lot': undefined,\n                    'active': true,\n                    'precision': {\n                        'amount': undefined,\n                        'price': undefined,\n                    },\n                    'limits': {\n                        'amount': {\n                            'min': undefined,\n                            'max': undefined,\n                        },\n                        'price': {\n                            'min': undefined,\n                            'max': undefined,\n                        },\n                        'cost': {\n                            'min': undefined,\n                            'max': undefined,\n                        },\n                    },\n                }));\n            }\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostInfoBalance (this.extend ({\n            'currency': 'ALL',\n        }, params));\n        let result = { 'info': response };\n        let balances = response['data'];\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            let lowercase = currency.toLowerCase ();\n            account['total'] = this.safeFloat (balances, 'total_' + lowercase);\n            account['used'] = this.safeFloat (balances, 'in_use_' + lowercase);\n            account['free'] = this.safeFloat (balances, 'available_' + lowercase);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetOrderbookCurrency (this.extend ({\n            'count': 50, // max = 50\n            'currency': market['base'],\n        }, params));\n        let orderbook = response['data'];\n        let timestamp = parseInt (orderbook['timestamp']);\n        return this.parseOrderBook (orderbook, timestamp, 'bids', 'asks', 'price', 'quantity');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = parseInt (ticker['date']);\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'max_price'),\n            'low': this.safeFloat (ticker, 'min_price'),\n            'bid': this.safeFloat (ticker, 'buy_price'),\n            'ask': this.safeFloat (ticker, 'sell_price'),\n            'vwap': undefined,\n            'open': this.safeFloat (ticker, 'opening_price'),\n            'close': this.safeFloat (ticker, 'closing_price'),\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last_trade'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': this.safeFloat (ticker, 'average_price'),\n            'baseVolume': this.safeFloat (ticker, 'volume_1day'),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetTickerAll (params);\n        let result = {};\n        let timestamp = response['data']['date'];\n        let tickers = this.omit (response['data'], 'date');\n        let ids = Object.keys (tickers);\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let symbol = id;\n            let market = undefined;\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            }\n            let ticker = tickers[id];\n            ticker['date'] = timestamp;\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTickerCurrency (this.extend ({\n            'currency': market['base'],\n        }, params));\n        return this.parseTicker (response['data'], market);\n    }\n\n    parseTrade (trade, market) {\n        // a workaround for their bug in date format, hours are not 0-padded\n        let [ transaction_date, transaction_time ] = trade['transaction_date'].split (' ');\n        let transaction_time_short = transaction_time.length < 8;\n        if (transaction_time_short)\n            transaction_time = '0' + transaction_time;\n        let timestamp = this.parse8601 (transaction_date + ' ' + transaction_time);\n        let side = (trade['type'] === 'ask') ? 'sell' : 'buy';\n        return {\n            'id': undefined,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': undefined,\n            'type': undefined,\n            'side': side,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['units_traded']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetRecentTransactionsCurrency (this.extend ({\n            'currency': market['base'],\n            'count': 100, // max = 100\n        }, params));\n        return this.parseTrades (response['data'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = undefined;\n        let method = 'privatePostTrade';\n        if (type === 'limit') {\n            request = {\n                'order_currency': market['id'],\n                'Payment_currency': market['quote'],\n                'units': amount,\n                'price': price,\n                'type': (side === 'buy') ? 'bid' : 'ask',\n            };\n            method += 'Place';\n        } else if (type === 'market') {\n            request = {\n                'currency': market['id'],\n                'units': amount,\n            };\n            method += 'Market' + this.capitalize (side);\n        }\n        let response = await this[method] (this.extend (request, params));\n        let id = undefined;\n        if ('order_id' in response) {\n            if (response['order_id'])\n                id = response['order_id'].toString ();\n        }\n        return {\n            'info': response,\n            'id': id,\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        let side = ('side' in params);\n        if (!side)\n            throw new ExchangeError (this.id + ' cancelOrder requires a side parameter (sell or buy) and a currency parameter');\n        side = (side === 'buy') ? 'purchase' : 'sales';\n        let currency = ('currency' in params);\n        if (!currency)\n            throw new ExchangeError (this.id + ' cancelOrder requires a currency parameter');\n        return await this.privatePostTradeCancel ({\n            'order_id': id,\n            'type': params['side'],\n            'currency': params['currency'],\n        });\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        let request = {\n            'units': amount,\n            'address': address,\n            'currency': currency,\n        };\n        if (currency === 'XRP' || currency === 'XMR') {\n            let destination = ('destination' in params);\n            if (!destination)\n                throw new ExchangeError (this.id + ' ' + currency + ' withdraw requires an extra destination param');\n        }\n        let response = await this.privatePostTradeBtcWithdrawal (this.extend (request, params));\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let endpoint = '/' + this.implodeParams (path, params);\n        let url = this.urls['api'][api] + endpoint;\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            body = this.urlencode (this.extend ({\n                'endpoint': endpoint,\n            }, query));\n            let nonce = this.nonce ().toString ();\n            let auth = endpoint + '\\0' + body + '\\0' + nonce;\n            let signature = this.hmac (this.encode (auth), this.encode (this.secret), 'sha512');\n            let signature64 = this.decode (this.stringToBase64 (this.encode (signature)));\n            headers = {\n                'Accept': 'application/json',\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Api-Key': this.apiKey,\n                'Api-Sign': signature64.toString (),\n                'Api-Nonce': nonce,\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('status' in response) {\n            if (response['status'] === '0000')\n                return response;\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { NotSupported } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitlish extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitlish',\n            'name': 'Bitlish',\n            'countries': [ 'GB', 'EU', 'RU' ],\n            'rateLimit': 1500,\n            'version': 'v1',\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n                'fetchOHLCV': true,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766275-dcfc6c30-5ed3-11e7-839d-00a846385d0b.jpg',\n                'api': 'https://bitlish.com/api',\n                'www': 'https://bitlish.com',\n                'doc': 'https://bitlish.com/api',\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': false,\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'taker': 0.3 / 100, // anonymous 0.3%, verified 0.2%\n                    'maker': 0,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.001,\n                        'LTC': 0.001,\n                        'DOGE': 0.001,\n                        'ETH': 0.001,\n                        'XMR': 0,\n                        'ZEC': 0.001,\n                        'DASH': 0.0001,\n                        'EUR': 50,\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                        'LTC': 0,\n                        'DOGE': 0,\n                        'ETH': 0,\n                        'XMR': 0,\n                        'ZEC': 0,\n                        'DASH': 0,\n                        'EUR': 0,\n                    },\n                },\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'instruments',\n                        'ohlcv',\n                        'pairs',\n                        'tickers',\n                        'trades_depth',\n                        'trades_history',\n                    ],\n                    'post': [\n                        'instruments',\n                        'ohlcv',\n                        'pairs',\n                        'tickers',\n                        'trades_depth',\n                        'trades_history',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'accounts_operations',\n                        'balance',\n                        'cancel_trade',\n                        'cancel_trades_by_ids',\n                        'cancel_all_trades',\n                        'create_bcode',\n                        'create_template_wallet',\n                        'create_trade',\n                        'deposit',\n                        'list_accounts_operations_from_ts',\n                        'list_active_trades',\n                        'list_bcodes',\n                        'list_my_matches_from_ts',\n                        'list_my_trades',\n                        'list_my_trads_from_ts',\n                        'list_payment_methods',\n                        'list_payments',\n                        'redeem_code',\n                        'resign',\n                        'signin',\n                        'signout',\n                        'trade_details',\n                        'trade_options',\n                        'withdraw',\n                        'withdraw_by_id',\n                    ],\n                },\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        if (!this.substituteCommonCurrencyCodes)\n            return currency;\n        if (currency === 'XBT')\n            return 'BTC';\n        if (currency === 'BCC')\n            return 'BCH';\n        if (currency === 'DRK')\n            return 'DASH';\n        if (currency === 'DSH')\n            currency = 'DASH';\n        if (currency === 'XDG')\n            currency = 'DOGE';\n        return currency;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetPairs ();\n        let result = [];\n        let keys = Object.keys (markets);\n        for (let p = 0; p < keys.length; p++) {\n            let market = markets[keys[p]];\n            let id = market['id'];\n            let symbol = market['name'];\n            let [ base, quote ] = symbol.split ('/');\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    parseTicker (ticker, market) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'high': this.safeFloat (ticker, 'max'),\n            'low': this.safeFloat (ticker, 'min'),\n            'bid': undefined,\n            'ask': undefined,\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': this.safeFloat (ticker, 'first'),\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': this.safeFloat (ticker, 'prc'),\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'sum'),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTickers (params);\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let tickers = await this.publicGetTickers (params);\n        let ticker = tickers[market['id']];\n        return this.parseTicker (ticker, market);\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        // let market = this.market (symbol);\n        let now = this.seconds ();\n        let start = now - 86400 * 30; // last 30 days\n        let interval = [ start.toString (), undefined ];\n        return await this.publicPostOhlcv (this.extend ({\n            'time_range': interval,\n        }, params));\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetTradesDepth (this.extend ({\n            'pair_id': this.marketId (symbol),\n        }, params));\n        let timestamp = undefined;\n        let last = this.safeInteger (orderbook, 'last');\n        if (last)\n            timestamp = parseInt (last / 1000);\n        return this.parseOrderBook (orderbook, timestamp, 'bid', 'ask', 'price', 'volume');\n    }\n\n    parseTrade (trade, market = undefined) {\n        let side = (trade['dir'] === 'bid') ? 'buy' : 'sell';\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = parseInt (trade['created'] / 1000);\n        return {\n            'id': undefined,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'order': undefined,\n            'type': undefined,\n            'side': side,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTradesHistory (this.extend ({\n            'pair_id': market['id'],\n        }, params));\n        return this.parseTrades (response['list'], market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostBalance ();\n        let result = { 'info': response };\n        let currencies = Object.keys (response);\n        let balance = {};\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let account = response[currency];\n            currency = currency.toUpperCase ();\n            // issue #4 bitlish names Dash as DSH, instead of DASH\n            if (currency === 'DSH')\n                currency = 'DASH';\n            if (currency === 'XDG')\n                currency = 'DOGE';\n            balance[currency] = account;\n        }\n        currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            if (currency in balance) {\n                account['free'] = parseFloat (balance[currency]['funds']);\n                account['used'] = parseFloat (balance[currency]['holded']);\n                account['total'] = this.sum (account['free'], account['used']);\n            }\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    signIn () {\n        return this.privatePostSignin ({\n            'login': this.login,\n            'passwd': this.password,\n        });\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'pair_id': this.marketId (symbol),\n            'dir': (side === 'buy') ? 'bid' : 'ask',\n            'amount': amount,\n        };\n        if (type === 'limit')\n            order['price'] = price;\n        let result = await this.privatePostCreateTrade (this.extend (order, params));\n        return {\n            'info': result,\n            'id': result['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelTrade ({ 'id': id });\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        if (currency !== 'BTC') {\n            // they did not document other types...\n            throw new NotSupported (this.id + ' currently supports BTC withdrawals only, until they document other currencies...');\n        }\n        let response = await this.privatePostWithdraw (this.extend ({\n            'currency': currency.toLowerCase (),\n            'amount': parseFloat (amount),\n            'account': address,\n            'payment_method': 'bitcoin', // they did not document other types...\n        }, params));\n        return {\n            'info': response,\n            'id': response['message_id'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + path;\n        if (api === 'public') {\n            if (method === 'GET') {\n                if (Object.keys (params).length)\n                    url += '?' + this.urlencode (params);\n            } else {\n                body = this.json (params);\n                headers = { 'Content-Type': 'application/json' };\n            }\n        } else {\n            this.checkRequiredCredentials ();\n            body = this.json (this.extend ({ 'token': this.apiKey }, params));\n            headers = { 'Content-Type': 'application/json' };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitmarket extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitmarket',\n            'name': 'BitMarket',\n            'countries': [ 'PL', 'EU' ],\n            'rateLimit': 1500,\n            'has': {\n                'CORS': false,\n                'fetchOHLCV': true,\n                'withdraw': true,\n            },\n            'timeframes': {\n                '90m': '90m',\n                '6h': '6h',\n                '1d': '1d',\n                '1w': '7d',\n                '1M': '1m',\n                '3M': '3m',\n                '6M': '6m',\n                '1y': '1y',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27767256-a8555200-5ef9-11e7-96fd-469a65e2b0bd.jpg',\n                'api': {\n                    'public': 'https://www.bitmarket.net',\n                    'private': 'https://www.bitmarket.pl/api2/', // last slash is critical\n                },\n                'www': [\n                    'https://www.bitmarket.pl',\n                    'https://www.bitmarket.net',\n                ],\n                'doc': [\n                    'https://www.bitmarket.net/docs.php?file=api_public.html',\n                    'https://www.bitmarket.net/docs.php?file=api_private.html',\n                    'https://github.com/bitmarket-net/api',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'json/{market}/ticker',\n                        'json/{market}/orderbook',\n                        'json/{market}/trades',\n                        'json/ctransfer',\n                        'graphs/{market}/90m',\n                        'graphs/{market}/6h',\n                        'graphs/{market}/1d',\n                        'graphs/{market}/7d',\n                        'graphs/{market}/1m',\n                        'graphs/{market}/3m',\n                        'graphs/{market}/6m',\n                        'graphs/{market}/1y',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'info',\n                        'trade',\n                        'cancel',\n                        'orders',\n                        'trades',\n                        'history',\n                        'withdrawals',\n                        'tradingdesk',\n                        'tradingdeskStatus',\n                        'tradingdeskConfirm',\n                        'cryptotradingdesk',\n                        'cryptotradingdeskStatus',\n                        'cryptotradingdeskConfirm',\n                        'withdraw',\n                        'withdrawFiat',\n                        'withdrawPLNPP',\n                        'withdrawFiatFast',\n                        'deposit',\n                        'transfer',\n                        'transfers',\n                        'marginList',\n                        'marginOpen',\n                        'marginClose',\n                        'marginCancel',\n                        'marginModify',\n                        'marginBalanceAdd',\n                        'marginBalanceRemove',\n                        'swapList',\n                        'swapOpen',\n                        'swapClose',\n                    ],\n                },\n            },\n            'markets': {\n                'BCH/PLN': { 'id': 'BCCPLN', 'symbol': 'BCH/PLN', 'base': 'BCH', 'quote': 'PLN' },\n                'BTG/PLN': { 'id': 'BTGPLN', 'symbol': 'BTG/PLN', 'base': 'BTG', 'quote': 'PLN' },\n                'BTC/PLN': { 'id': 'BTCPLN', 'symbol': 'BTC/PLN', 'base': 'BTC', 'quote': 'PLN' },\n                'BTC/EUR': { 'id': 'BTCEUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR' },\n                'LTC/PLN': { 'id': 'LTCPLN', 'symbol': 'LTC/PLN', 'base': 'LTC', 'quote': 'PLN' },\n                'LTC/BTC': { 'id': 'LTCBTC', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC' },\n                'LiteMineX/BTC': { 'id': 'LiteMineXBTC', 'symbol': 'LiteMineX/BTC', 'base': 'LiteMineX', 'quote': 'BTC' },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': true,\n                    'percentage': true,\n                    'taker': 0.45 / 100,\n                    'maker': 0.15 / 100,\n                    'tiers': {\n                        'taker': [\n                            [0, 0.45 / 100],\n                            [99.99, 0.44 / 100],\n                            [299.99, 0.43 / 100],\n                            [499.99, 0.42 / 100],\n                            [999.99, 0.41 / 100],\n                            [1999.99, 0.40 / 100],\n                            [2999.99, 0.39 / 100],\n                            [4999.99, 0.38 / 100],\n                            [9999.99, 0.37 / 100],\n                            [19999.99, 0.36 / 100],\n                            [29999.99, 0.35 / 100],\n                            [49999.99, 0.34 / 100],\n                            [99999.99, 0.33 / 100],\n                            [199999.99, 0.32 / 100],\n                            [299999.99, 0.31 / 100],\n                            [499999.99, 0.0 / 100],\n                        ],\n                        'maker': [\n                            [0, 0.15 / 100],\n                            [99.99, 0.14 / 100],\n                            [299.99, 0.13 / 100],\n                            [499.99, 0.12 / 100],\n                            [999.99, 0.11 / 100],\n                            [1999.99, 0.10 / 100],\n                            [2999.99, 0.9 / 100],\n                            [4999.99, 0.8 / 100],\n                            [9999.99, 0.7 / 100],\n                            [19999.99, 0.6 / 100],\n                            [29999.99, 0.5 / 100],\n                            [49999.99, 0.4 / 100],\n                            [99999.99, 0.3 / 100],\n                            [199999.99, 0.2 / 100],\n                            [299999.99, 0.1 / 100],\n                            [499999.99, 0.0 / 100],\n                        ],\n                    },\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.0008,\n                        'LTC': 0.005,\n                        'BCH': 0.0008,\n                        'BTG': 0.0008,\n                        'DOGE': 1,\n                        'EUR': 2,\n                        'PLN': 2,\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                        'LTC': 0,\n                        'BCH': 0,\n                        'BTG': 0,\n                        'DOGE': 25,\n                        'EUR': 2, // SEPA. Transfer INT (SHA): 5 EUR\n                        'PLN': 0,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostInfo ();\n        let data = response['data'];\n        let balance = data['balances'];\n        let result = { 'info': data };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            if (currency in balance['available'])\n                account['free'] = balance['available'][currency];\n            if (currency in balance['blocked'])\n                account['used'] = balance['blocked'][currency];\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetJsonMarketOrderbook (this.extend ({\n            'market': this.marketId (symbol),\n        }, params));\n        let timestamp = this.milliseconds ();\n        return {\n            'bids': orderbook['bids'],\n            'asks': orderbook['asks'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetJsonMarketTicker (this.extend ({\n            'market': this.marketId (symbol),\n        }, params));\n        let timestamp = this.milliseconds ();\n        let vwap = parseFloat (ticker['vwap']);\n        let baseVolume = parseFloat (ticker['volume']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': vwap,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let side = (trade['type'] === 'bid') ? 'buy' : 'sell';\n        let timestamp = trade['date'] * 1000;\n        return {\n            'id': trade['tid'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': undefined,\n            'type': undefined,\n            'side': side,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetJsonMarketTrades (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '90m', since = undefined, limit = undefined) {\n        return [\n            ohlcv['time'] * 1000,\n            parseFloat (ohlcv['open']),\n            parseFloat (ohlcv['high']),\n            parseFloat (ohlcv['low']),\n            parseFloat (ohlcv['close']),\n            parseFloat (ohlcv['vol']),\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '90m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let method = 'publicGetGraphsMarket' + this.timeframes[timeframe];\n        let market = this.market (symbol);\n        let response = await this[method] (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let response = await this.privatePostTrade (this.extend ({\n            'market': this.marketId (symbol),\n            'type': side,\n            'amount': amount,\n            'rate': price,\n        }, params));\n        let result = {\n            'info': response,\n        };\n        if ('id' in response['order'])\n            result['id'] = response['id'];\n        return result;\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancel ({ 'id': id });\n    }\n\n    isFiat (currency) {\n        if (currency === 'EUR')\n            return true;\n        if (currency === 'PLN')\n            return true;\n        return false;\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let method = undefined;\n        let request = {\n            'currency': currency,\n            'quantity': amount,\n        };\n        if (this.isFiat (currency)) {\n            method = 'privatePostWithdrawFiat';\n            if ('account' in params) {\n                request['account'] = params['account']; // bank account code for withdrawal\n            } else {\n                throw new ExchangeError (this.id + ' requires account parameter to withdraw fiat currency');\n            }\n            if ('account2' in params) {\n                request['account2'] = params['account2']; // bank SWIFT code (EUR only)\n            } else {\n                if (currency === 'EUR')\n                    throw new ExchangeError (this.id + ' requires account2 parameter to withdraw EUR');\n            }\n            if ('withdrawal_note' in params) {\n                request['withdrawal_note'] = params['withdrawal_note']; // a 10-character user-specified withdrawal note (PLN only)\n            } else {\n                if (currency === 'PLN')\n                    throw new ExchangeError (this.id + ' requires withdrawal_note parameter to withdraw PLN');\n            }\n        } else {\n            method = 'privatePostWithdraw';\n            request['address'] = address;\n        }\n        let response = await this[method] (this.extend (request, params));\n        return {\n            'info': response,\n            'id': response,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        if (api === 'public') {\n            url += '/' + this.implodeParams (path + '.json', params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let query = this.extend ({\n                'tonce': nonce,\n                'method': path,\n            }, params);\n            body = this.urlencode (query);\n            headers = {\n                'API-Key': this.apiKey,\n                'API-Hash': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, DDoSProtection } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitmex extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitmex',\n            'name': 'BitMEX',\n            'countries': 'SC', // Seychelles\n            'version': 'v1',\n            'userAgent': undefined,\n            'rateLimit': 1500,\n            'has': {\n                'CORS': false,\n                'fetchOHLCV': true,\n                'withdraw': true,\n            },\n            'timeframes': {\n                '1m': '1m',\n                '5m': '5m',\n                '1h': '1h',\n                '1d': '1d',\n            },\n            'urls': {\n                'test': 'https://testnet.bitmex.com',\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766319-f653c6e6-5ed4-11e7-933d-f0bc3699ae8f.jpg',\n                'api': 'https://www.bitmex.com',\n                'www': 'https://www.bitmex.com',\n                'doc': [\n                    'https://www.bitmex.com/app/apiOverview',\n                    'https://github.com/BitMEX/api-connectors/tree/master/official-http',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'announcement',\n                        'announcement/urgent',\n                        'funding',\n                        'instrument',\n                        'instrument/active',\n                        'instrument/activeAndIndices',\n                        'instrument/activeIntervals',\n                        'instrument/compositeIndex',\n                        'instrument/indices',\n                        'insurance',\n                        'leaderboard',\n                        'liquidation',\n                        'orderBook',\n                        'orderBook/L2',\n                        'quote',\n                        'quote/bucketed',\n                        'schema',\n                        'schema/websocketHelp',\n                        'settlement',\n                        'stats',\n                        'stats/history',\n                        'trade',\n                        'trade/bucketed',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'apiKey',\n                        'chat',\n                        'chat/channels',\n                        'chat/connected',\n                        'execution',\n                        'execution/tradeHistory',\n                        'notification',\n                        'order',\n                        'position',\n                        'user',\n                        'user/affiliateStatus',\n                        'user/checkReferralCode',\n                        'user/commission',\n                        'user/depositAddress',\n                        'user/margin',\n                        'user/minWithdrawalFee',\n                        'user/wallet',\n                        'user/walletHistory',\n                        'user/walletSummary',\n                    ],\n                    'post': [\n                        'apiKey',\n                        'apiKey/disable',\n                        'apiKey/enable',\n                        'chat',\n                        'order',\n                        'order/bulk',\n                        'order/cancelAllAfter',\n                        'order/closePosition',\n                        'position/isolate',\n                        'position/leverage',\n                        'position/riskLimit',\n                        'position/transferMargin',\n                        'user/cancelWithdrawal',\n                        'user/confirmEmail',\n                        'user/confirmEnableTFA',\n                        'user/confirmWithdrawal',\n                        'user/disableTFA',\n                        'user/logout',\n                        'user/logoutAll',\n                        'user/preferences',\n                        'user/requestEnableTFA',\n                        'user/requestWithdrawal',\n                    ],\n                    'put': [\n                        'order',\n                        'order/bulk',\n                        'user',\n                    ],\n                    'delete': [\n                        'apiKey',\n                        'order',\n                        'order/all',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetInstrumentActiveAndIndices ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let active = (market['state'] !== 'Unlisted');\n            let id = market['symbol'];\n            let base = market['underlying'];\n            let quote = market['quoteCurrency'];\n            let type = undefined;\n            let future = false;\n            let prediction = false;\n            let basequote = base + quote;\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let swap = (id === basequote);\n            let symbol = id;\n            if (swap) {\n                type = 'swap';\n                symbol = base + '/' + quote;\n            } else if (id.indexOf ('B_') >= 0) {\n                prediction = true;\n                type = 'prediction';\n            } else {\n                future = true;\n                type = 'future';\n            }\n            let maker = market['makerFee'];\n            let taker = market['takerFee'];\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': active,\n                'taker': taker,\n                'maker': maker,\n                'type': type,\n                'spot': false,\n                'swap': swap,\n                'future': future,\n                'prediction': prediction,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetUserMargin ({ 'currency': 'all' });\n        let result = { 'info': response };\n        for (let b = 0; b < response.length; b++) {\n            let balance = response[b];\n            let currency = balance['currency'].toUpperCase ();\n            currency = this.commonCurrencyCode (currency);\n            let account = {\n                'free': balance['availableMargin'],\n                'used': 0.0,\n                'total': balance['marginBalance'],\n            };\n            if (currency === 'BTC') {\n                account['free'] = account['free'] * 0.00000001;\n                account['total'] = account['total'] * 0.00000001;\n            }\n            account['used'] = account['total'] - account['free'];\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderBookL2 (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        let timestamp = this.milliseconds ();\n        let result = {\n            'bids': [],\n            'asks': [],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n        };\n        for (let o = 0; o < orderbook.length; o++) {\n            let order = orderbook[o];\n            let side = (order['side'] === 'Sell') ? 'asks' : 'bids';\n            let amount = order['size'];\n            let price = order['price'];\n            result[side].push ([ price, amount ]);\n        }\n        result['bids'] = this.sortBy (result['bids'], 0, true);\n        result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        if (!market['active'])\n            throw new ExchangeError (this.id + ': symbol ' + symbol + ' is delisted');\n        let request = this.extend ({\n            'symbol': market['id'],\n            'binSize': '1d',\n            'partial': true,\n            'count': 1,\n            'reverse': true,\n        }, params);\n        let quotes = await this.publicGetQuoteBucketed (request);\n        let quotesLength = quotes.length;\n        let quote = quotes[quotesLength - 1];\n        let tickers = await this.publicGetTradeBucketed (request);\n        let ticker = tickers[0];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (quote['bidPrice']),\n            'ask': parseFloat (quote['askPrice']),\n            'vwap': parseFloat (ticker['vwap']),\n            'open': undefined,\n            'close': parseFloat (ticker['close']),\n            'first': undefined,\n            'last': undefined,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['homeNotional']),\n            'quoteVolume': parseFloat (ticker['foreignNotional']),\n            'info': ticker,\n        };\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        let timestamp = this.parse8601 (ohlcv['timestamp']);\n        return [\n            timestamp,\n            ohlcv['open'],\n            ohlcv['high'],\n            ohlcv['low'],\n            ohlcv['close'],\n            ohlcv['volume'],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        // send JSON key/value pairs, such as {\"key\": \"value\"}\n        // filter by individual fields and do advanced queries on timestamps\n        // let filter = { 'key': 'value' };\n        // send a bare series (e.g. XBU) to nearest expiring contract in that series\n        // you can also send a timeframe, e.g. XBU:monthly\n        // timeframes: daily, weekly, monthly, quarterly, and biquarterly\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n            'binSize': this.timeframes[timeframe],\n            'partial': true,     // true == include yet-incomplete current bins\n            // 'filter': filter, // filter by individual fields and do advanced queries\n            // 'columns': [],    // will return all columns if omitted\n            // 'start': 0,       // starting point for results (wtf?)\n            // 'reverse': false, // true == newest first\n            // 'endTime': '',    // ending date filter for results\n        };\n        if (since) {\n            let ymdhms = this.YmdHMS (since);\n            let ymdhm = ymdhms.slice (0, 16);\n            request['startTime'] = ymdhm; // starting date filter for results\n        }\n        if (limit)\n            request['count'] = limit; // default 100\n        let response = await this.publicGetTradeBucketed (this.extend (request, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = this.parse8601 (trade['timestamp']);\n        let symbol = undefined;\n        if (!market) {\n            if ('symbol' in trade)\n                market = this.markets_by_id[trade['symbol']];\n        }\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'id': trade['trdMatchID'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'order': undefined,\n            'type': undefined,\n            'side': trade['side'].toLowerCase (),\n            'price': trade['price'],\n            'amount': trade['size'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTrade (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'symbol': this.marketId (symbol),\n            'side': this.capitalize (side),\n            'orderQty': amount,\n            'ordType': this.capitalize (type),\n        };\n        if (type === 'limit')\n            order['price'] = price;\n        let response = await this.privatePostOrder (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['orderID'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privateDeleteOrder ({ 'orderID': id });\n    }\n\n    isFiat (currency) {\n        if (currency === 'EUR')\n            return true;\n        if (currency === 'PLN')\n            return true;\n        return false;\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        if (currency !== 'BTC')\n            throw new ExchangeError (this.id + ' supoprts BTC withdrawals only, other currencies coming soon...');\n        let request = {\n            'currency': 'XBt', // temporarily\n            'amount': amount,\n            'address': address,\n            // 'otpToken': '123456', // requires if two-factor auth (OTP) is enabled\n            // 'fee': 0.001, // bitcoin network fee\n        };\n        let response = await this.privatePostUserRequestWithdrawal (this.extend (request, params));\n        return {\n            'info': response,\n            'id': response['transactID'],\n        };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code === 429)\n            throw new DDoSProtection (this.id + ' ' + body);\n        if (code >= 400) {\n            if (body) {\n                if (body[0] === '{') {\n                    let response = JSON.parse (body);\n                    if ('error' in response) {\n                        if ('message' in response['error']) {\n                            // stub code, need proper handling\n                            throw new ExchangeError (this.id + ' ' + this.json (response));\n                        }\n                    }\n                }\n            }\n        }\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let query = '/api' + '/' + this.version + '/' + path;\n        if (method !== 'PUT')\n            if (Object.keys (params).length)\n                query += '?' + this.urlencode (params);\n        let url = this.urls['api'] + query;\n        if (api === 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let auth = method + query + nonce;\n            if (method === 'POST' || method === 'PUT') {\n                if (Object.keys (params).length) {\n                    body = this.json (params);\n                    auth += body;\n                }\n            }\n            headers = {\n                'Content-Type': 'application/json',\n                'api-nonce': nonce,\n                'api-key': this.apiKey,\n                'api-signature': this.hmac (this.encode (auth), this.encode (this.secret)),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitso extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitso',\n            'name': 'Bitso',\n            'countries': 'MX', // Mexico\n            'rateLimit': 2000, // 30 requests per minute\n            'version': 'v3',\n            'has': {\n                'CORS': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766335-715ce7aa-5ed5-11e7-88a8-173a27bb30fe.jpg',\n                'api': 'https://api.bitso.com',\n                'www': 'https://bitso.com',\n                'doc': 'https://bitso.com/api_info',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'available_books',\n                        'ticker',\n                        'order_book',\n                        'trades',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'account_status',\n                        'balance',\n                        'fees',\n                        'fundings',\n                        'fundings/{fid}',\n                        'funding_destination',\n                        'kyc_documents',\n                        'ledger',\n                        'ledger/trades',\n                        'ledger/fees',\n                        'ledger/fundings',\n                        'ledger/withdrawals',\n                        'mx_bank_codes',\n                        'open_orders',\n                        'order_trades/{oid}',\n                        'orders/{oid}',\n                        'user_trades',\n                        'user_trades/{tid}',\n                        'withdrawals/',\n                        'withdrawals/{wid}',\n                    ],\n                    'post': [\n                        'bitcoin_withdrawal',\n                        'debit_card_withdrawal',\n                        'ether_withdrawal',\n                        'orders',\n                        'phone_number',\n                        'phone_verification',\n                        'phone_withdrawal',\n                        'spei_withdrawal',\n                    ],\n                    'delete': [\n                        'orders/{oid}',\n                        'orders/all',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetAvailableBooks ();\n        let result = [];\n        for (let i = 0; i < markets['payload'].length; i++) {\n            let market = markets['payload'][i];\n            let id = market['book'];\n            let symbol = id.toUpperCase ().replace ('_', '/');\n            let [ base, quote ] = symbol.split ('/');\n            let limits = {\n                'amount': {\n                    'min': parseFloat (market['minimum_amount']),\n                    'max': parseFloat (market['maximum_amount']),\n                },\n                'price': {\n                    'min': parseFloat (market['minimum_price']),\n                    'max': parseFloat (market['maximum_price']),\n                },\n                'cost': {\n                    'min': parseFloat (market['minimum_value']),\n                    'max': parseFloat (market['maximum_value']),\n                },\n            };\n            let precision = {\n                'amount': this.precisionFromString (market['minimum_amount']),\n                'price': this.precisionFromString (market['minimum_price']),\n            };\n            let lot = limits['amount']['min'];\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n                'lot': lot,\n                'limits': limits,\n                'precision': precision,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetBalance ();\n        let balances = response['payload']['balances'];\n        let result = { 'info': response };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'].toUpperCase ();\n            let account = {\n                'free': parseFloat (balance['available']),\n                'used': parseFloat (balance['locked']),\n                'total': parseFloat (balance['total']),\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetOrderBook (this.extend ({\n            'book': this.marketId (symbol),\n        }, params));\n        let orderbook = response['payload'];\n        let timestamp = this.parse8601 (orderbook['updated_at']);\n        return this.parseOrderBook (orderbook, timestamp, 'bids', 'asks', 'price', 'amount');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetTicker (this.extend ({\n            'book': this.marketId (symbol),\n        }, params));\n        let ticker = response['payload'];\n        let timestamp = this.parse8601 (ticker['created_at']);\n        let vwap = parseFloat (ticker['vwap']);\n        let baseVolume = parseFloat (ticker['volume']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': vwap,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = this.parse8601 (trade['created_at']);\n        let symbol = undefined;\n        if (!market) {\n            if ('book' in trade)\n                market = this.markets_by_id[trade['book']];\n        }\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'id': trade['tid'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'order': undefined,\n            'type': undefined,\n            'side': trade['maker_side'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTrades (this.extend ({\n            'book': market['id'],\n        }, params));\n        return this.parseTrades (response['payload'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'book': this.marketId (symbol),\n            'side': side,\n            'type': type,\n            'major': this.amountToPrecision (symbol, amount),\n        };\n        if (type === 'limit')\n            order['price'] = this.priceToPrecision (symbol, price);\n        let response = await this.privatePostOrders (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['payload']['oid'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privateDeleteOrders ({ 'oid': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let query = '/' + this.version + '/' + this.implodeParams (path, params);\n        let url = this.urls['api'] + query;\n        if (api === 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let request = [ nonce, method, query ].join ('');\n            if (Object.keys (params).length) {\n                body = this.json (params);\n                request += body;\n            }\n            let signature = this.hmac (this.encode (request), this.encode (this.secret));\n            let auth = this.apiKey + ':' + nonce + ':' + signature;\n            headers = {\n                'Authorization': 'Bitso ' + auth,\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('success' in response)\n            if (response['success'])\n                return response;\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitstamp extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitstamp',\n            'name': 'Bitstamp',\n            'countries': 'GB',\n            'rateLimit': 1000,\n            'version': 'v2',\n            'has': {\n                'CORS': true,\n                'fetchOrder': true,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27786377-8c8ab57e-5fe9-11e7-8ea4-2b05b6bcceec.jpg',\n                'api': 'https://www.bitstamp.net/api',\n                'www': 'https://www.bitstamp.net',\n                'doc': 'https://www.bitstamp.net/api',\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'uid': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'order_book/{pair}/',\n                        'ticker_hour/{pair}/',\n                        'ticker/{pair}/',\n                        'transactions/{pair}/',\n                        'trading-pairs-info/',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balance/',\n                        'balance/{pair}/',\n                        'bch_withdrawal/',\n                        'bch_address/',\n                        'user_transactions/',\n                        'user_transactions/{pair}/',\n                        'open_orders/all/',\n                        'open_orders/{pair}/',\n                        'order_status/',\n                        'cancel_order/',\n                        'buy/{pair}/',\n                        'buy/market/{pair}/',\n                        'sell/{pair}/',\n                        'sell/market/{pair}/',\n                        'ltc_withdrawal/',\n                        'ltc_address/',\n                        'eth_withdrawal/',\n                        'eth_address/',\n                        'xrp_withdrawal/',\n                        'xrp_address/',\n                        'transfer-to-main/',\n                        'transfer-from-main/',\n                        'withdrawal/open/',\n                        'withdrawal/status/',\n                        'withdrawal/cancel/',\n                        'liquidation_address/new/',\n                        'liquidation_address/info/',\n                    ],\n                },\n                'v1': {\n                    'post': [\n                        'bitcoin_deposit_address/',\n                        'unconfirmed_btc/',\n                        'bitcoin_withdrawal/',\n                        'ripple_withdrawal/',\n                        'ripple_address/',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': true,\n                    'percentage': true,\n                    'taker': 0.25 / 100,\n                    'maker': 0.25 / 100,\n                    'tiers': {\n                        'taker': [\n                            [0, 0.25 / 100],\n                            [20000, 0.24 / 100],\n                            [100000, 0.22 / 100],\n                            [400000, 0.20 / 100],\n                            [600000, 0.15 / 100],\n                            [1000000, 0.14 / 100],\n                            [2000000, 0.13 / 100],\n                            [4000000, 0.12 / 100],\n                            [20000000, 0.11 / 100],\n                            [20000001, 0.10 / 100],\n                        ],\n                        'maker': [\n                            [0, 0.25 / 100],\n                            [20000, 0.24 / 100],\n                            [100000, 0.22 / 100],\n                            [400000, 0.20 / 100],\n                            [600000, 0.15 / 100],\n                            [1000000, 0.14 / 100],\n                            [2000000, 0.13 / 100],\n                            [4000000, 0.12 / 100],\n                            [20000000, 0.11 / 100],\n                            [20000001, 0.10 / 100],\n                        ],\n                    },\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0,\n                        'BCH': 0,\n                        'LTC': 0,\n                        'ETH': 0,\n                        'XRP': 0,\n                        'USD': 25,\n                        'EUR': 0.90,\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                        'BCH': 0,\n                        'LTC': 0,\n                        'ETH': 0,\n                        'XRP': 0,\n                        'USD': 25,\n                        'EUR': 0,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetTradingPairsInfo ();\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let symbol = market['name'];\n            let [ base, quote ] = symbol.split ('/');\n            let baseId = base.toLowerCase ();\n            let quoteId = quote.toLowerCase ();\n            let symbolId = baseId + '_' + quoteId;\n            let id = market['url_symbol'];\n            let precision = {\n                'amount': market['base_decimals'],\n                'price': market['counter_decimals'],\n            };\n            let [ cost, currency ] = market['minimum_order'].split (' ');\n            let active = (market['trading'] === 'Enabled');\n            let lot = Math.pow (10, -precision['amount']);\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'baseId': baseId,\n                'quoteId': quoteId,\n                'symbolId': symbolId,\n                'info': market,\n                'lot': lot,\n                'active': active,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision['price']),\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': parseFloat (cost),\n                        'max': undefined,\n                    },\n                },\n            });\n        }\n        return result;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderBookPair (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        let timestamp = parseInt (orderbook['timestamp']) * 1000;\n        return this.parseOrderBook (orderbook, timestamp);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let ticker = await this.publicGetTickerPair (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        let timestamp = parseInt (ticker['timestamp']) * 1000;\n        let vwap = parseFloat (ticker['vwap']);\n        let baseVolume = parseFloat (ticker['volume']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': vwap,\n            'open': parseFloat (ticker['open']),\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = undefined;\n        if ('date' in trade) {\n            timestamp = parseInt (trade['date']) * 1000;\n        } else if ('datetime' in trade) {\n            timestamp = this.parse8601 (trade['datetime']);\n        }\n        let side = (trade['type'] === '0') ? 'buy' : 'sell';\n        let order = undefined;\n        if ('order_id' in trade)\n            order = trade['order_id'].toString ();\n        if ('currency_pair' in trade) {\n            let marketId = trade['currency_pair'];\n            if (marketId in this.markets_by_id)\n                market = this.markets_by_id[marketId];\n        }\n        let price = this.safeFloat (trade, 'price');\n        price = this.safeFloat (trade, market['symbolId'], price);\n        let amount = this.safeFloat (trade, 'amount');\n        amount = this.safeFloat (trade, market['baseId'], amount);\n        let id = this.safeValue (trade, 'tid');\n        id = this.safeValue (trade, 'id', id);\n        if (id)\n            id = id.toString ();\n        return {\n            'id': id,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': order,\n            'type': undefined,\n            'side': side,\n            'price': parseFloat (price),\n            'amount': parseFloat (amount),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTransactionsPair (this.extend ({\n            'pair': market['id'],\n            'time': 'minute',\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balance = await this.privatePostBalance ();\n        let result = { 'info': balance };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let total = lowercase + '_balance';\n            let free = lowercase + '_available';\n            let used = lowercase + '_reserved';\n            let account = this.account ();\n            if (free in balance)\n                account['free'] = parseFloat (balance[free]);\n            if (used in balance)\n                account['used'] = parseFloat (balance[used]);\n            if (total in balance)\n                account['total'] = parseFloat (balance[total]);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let method = 'privatePost' + this.capitalize (side);\n        let order = {\n            'pair': this.marketId (symbol),\n            'amount': amount,\n        };\n        if (type === 'market')\n            method += 'Market';\n        else\n            order['price'] = price;\n        method += 'Pair';\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelOrder ({ 'id': id });\n    }\n\n    parseOrderStatus (order) {\n        if ((order['status'] === 'Queue') || (order['status'] === 'Open'))\n            return 'open';\n        if (order['status'] === 'Finished')\n            return 'closed';\n        return order['status'];\n    }\n\n    async fetchOrderStatus (id, symbol = undefined) {\n        await this.loadMarkets ();\n        let response = await this.privatePostOrderStatus ({ 'id': id });\n        return this.parseOrderStatus (response);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        if (symbol)\n            market = this.market (symbol);\n        let pair = market ? market['id'] : 'all';\n        let request = this.extend ({ 'pair': pair }, params);\n        let response = await this.privatePostUserTransactionsPair (request);\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostOrderStatus ({ 'id': id });\n    }\n\n    getCurrencyName (code) {\n        if (code === 'BTC')\n            return 'bitcoin';\n        return code.toLowerCase ();\n    }\n\n    isFiat (code) {\n        if (code === 'USD')\n            return true;\n        if (code === 'EUR')\n            return true;\n        return false;\n    }\n\n    async withdraw (code, amount, address, tag = undefined, params = {}) {\n        let isFiat = this.isFiat (code);\n        if (isFiat)\n            throw new ExchangeError (this.id + ' fiat withdraw() for ' + code + ' is not implemented yet');\n        let name = this.getCurrencyName (code);\n        let request = {\n            'amount': amount,\n            'address': address,\n        };\n        let v1 = (code === 'BTC');\n        let method = v1 ? 'v1' : 'private'; // v1 or v2\n        method += 'Post' + this.capitalize (name) + 'Withdrawal';\n        let query = params;\n        if (code === 'XRP') {\n            let tag = this.safeString (params, 'destination_tag');\n            if (tag) {\n                request['destination_tag'] = tag;\n                query = this.omit (params, 'destination_tag');\n            } else {\n                throw new ExchangeError (this.id + ' withdraw() requires a destination_tag param for ' + code);\n            }\n        }\n        let response = await this[method] (this.extend (request, query));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/';\n        if (api !== 'v1')\n            url += this.version + '/';\n        url += this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let auth = nonce + this.uid + this.apiKey;\n            let signature = this.encode (this.hmac (this.encode (auth), this.encode (this.secret)));\n            query = this.extend ({\n                'key': this.apiKey,\n                'signature': signature.toUpperCase (),\n                'nonce': nonce,\n            }, query);\n            body = this.urlencode (query);\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('status' in response)\n            if (response['status'] === 'error')\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, NotSupported } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bitstamp1 extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bitstamp1',\n            'name': 'Bitstamp v1',\n            'countries': 'GB',\n            'rateLimit': 1000,\n            'version': 'v1',\n            'has': {\n                'CORS': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27786377-8c8ab57e-5fe9-11e7-8ea4-2b05b6bcceec.jpg',\n                'api': 'https://www.bitstamp.net/api',\n                'www': 'https://www.bitstamp.net',\n                'doc': 'https://www.bitstamp.net/api',\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'uid': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'ticker',\n                        'ticker_hour',\n                        'order_book',\n                        'transactions',\n                        'eur_usd',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balance',\n                        'user_transactions',\n                        'open_orders',\n                        'order_status',\n                        'cancel_order',\n                        'cancel_all_orders',\n                        'buy',\n                        'sell',\n                        'bitcoin_deposit_address',\n                        'unconfirmed_btc',\n                        'ripple_withdrawal',\n                        'ripple_address',\n                        'withdrawal_requests',\n                        'bitcoin_withdrawal',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/USD': { 'id': 'btcusd', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD', 'maker': 0.0025, 'taker': 0.0025 },\n                'BTC/EUR': { 'id': 'btceur', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR', 'maker': 0.0025, 'taker': 0.0025 },\n                'EUR/USD': { 'id': 'eurusd', 'symbol': 'EUR/USD', 'base': 'EUR', 'quote': 'USD', 'maker': 0.0025, 'taker': 0.0025 },\n                'XRP/USD': { 'id': 'xrpusd', 'symbol': 'XRP/USD', 'base': 'XRP', 'quote': 'USD', 'maker': 0.0025, 'taker': 0.0025 },\n                'XRP/EUR': { 'id': 'xrpeur', 'symbol': 'XRP/EUR', 'base': 'XRP', 'quote': 'EUR', 'maker': 0.0025, 'taker': 0.0025 },\n                'XRP/BTC': { 'id': 'xrpbtc', 'symbol': 'XRP/BTC', 'base': 'XRP', 'quote': 'BTC', 'maker': 0.0025, 'taker': 0.0025 },\n                'LTC/USD': { 'id': 'ltcusd', 'symbol': 'LTC/USD', 'base': 'LTC', 'quote': 'USD', 'maker': 0.0025, 'taker': 0.0025 },\n                'LTC/EUR': { 'id': 'ltceur', 'symbol': 'LTC/EUR', 'base': 'LTC', 'quote': 'EUR', 'maker': 0.0025, 'taker': 0.0025 },\n                'LTC/BTC': { 'id': 'ltcbtc', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC', 'maker': 0.0025, 'taker': 0.0025 },\n                'ETH/USD': { 'id': 'ethusd', 'symbol': 'ETH/USD', 'base': 'ETH', 'quote': 'USD', 'maker': 0.0025, 'taker': 0.0025 },\n                'ETH/EUR': { 'id': 'etheur', 'symbol': 'ETH/EUR', 'base': 'ETH', 'quote': 'EUR', 'maker': 0.0025, 'taker': 0.0025 },\n                'ETH/BTC': { 'id': 'ethbtc', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC', 'maker': 0.0025, 'taker': 0.0025 },\n            },\n        });\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        if (symbol !== 'BTC/USD')\n            throw new ExchangeError (this.id + ' ' + this.version + \" fetchOrderBook doesn't support \" + symbol + ', use it for BTC/USD only');\n        let orderbook = await this.publicGetOrderBook (params);\n        let timestamp = parseInt (orderbook['timestamp']) * 1000;\n        return this.parseOrderBook (orderbook, timestamp);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        if (symbol !== 'BTC/USD')\n            throw new ExchangeError (this.id + ' ' + this.version + \" fetchTicker doesn't support \" + symbol + ', use it for BTC/USD only');\n        let ticker = await this.publicGetTicker (params);\n        let timestamp = parseInt (ticker['timestamp']) * 1000;\n        let vwap = parseFloat (ticker['vwap']);\n        let baseVolume = parseFloat (ticker['volume']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': vwap,\n            'open': parseFloat (ticker['open']),\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = undefined;\n        if ('date' in trade) {\n            timestamp = parseInt (trade['date']) * 1000;\n        } else if ('datetime' in trade) {\n            // timestamp = this.parse8601 (trade['datetime']);\n            timestamp = parseInt (trade['datetime']) * 1000;\n        }\n        let side = (trade['type'] === 0) ? 'buy' : 'sell';\n        let order = undefined;\n        if ('order_id' in trade)\n            order = trade['order_id'].toString ();\n        if ('currency_pair' in trade) {\n            if (trade['currency_pair'] in this.markets_by_id)\n                market = this.markets_by_id[trade['currency_pair']];\n        }\n        return {\n            'id': trade['tid'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': order,\n            'type': undefined,\n            'side': side,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        if (symbol !== 'BTC/USD')\n            throw new ExchangeError (this.id + ' ' + this.version + \" fetchTrades doesn't support \" + symbol + ', use it for BTC/USD only');\n        let market = this.market (symbol);\n        let response = await this.publicGetTransactions (this.extend ({\n            'time': 'minute',\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        let balance = await this.privatePostBalance ();\n        let result = { 'info': balance };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let total = lowercase + '_balance';\n            let free = lowercase + '_available';\n            let used = lowercase + '_reserved';\n            let account = this.account ();\n            account['free'] = this.safeFloat (balance, free, 0.0);\n            account['used'] = this.safeFloat (balance, used, 0.0);\n            account['total'] = this.safeFloat (balance, total, 0.0);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type !== 'limit')\n            throw new ExchangeError (this.id + ' ' + this.version + ' accepts limit orders only');\n        if (symbol !== 'BTC/USD')\n            throw new ExchangeError (this.id + ' v1 supports BTC/USD orders only');\n        let method = 'privatePost' + this.capitalize (side);\n        let order = {\n            'amount': amount,\n            'price': price,\n        };\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelOrder ({ 'id': id });\n    }\n\n    parseOrderStatus (order) {\n        if ((order['status'] === 'Queue') || (order['status'] === 'Open'))\n            return 'open';\n        if (order['status'] === 'Finished')\n            return 'closed';\n        return order['status'];\n    }\n\n    async fetchOrderStatus (id, symbol = undefined) {\n        await this.loadMarkets ();\n        let response = await this.privatePostOrderStatus ({ 'id': id });\n        return this.parseOrderStatus (response);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        if (symbol)\n            market = this.market (symbol);\n        let pair = market ? market['id'] : 'all';\n        let request = this.extend ({ 'id': pair }, params);\n        let response = await this.privatePostOpenOrdersId (request);\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchOrder is not implemented yet');\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let auth = nonce + this.uid + this.apiKey;\n            let signature = this.encode (this.hmac (this.encode (auth), this.encode (this.secret)));\n            query = this.extend ({\n                'key': this.apiKey,\n                'signature': signature.toUpperCase (),\n                'nonce': nonce,\n            }, query);\n            body = this.urlencode (query);\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('status' in response)\n            if (response['status'] === 'error')\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, AuthenticationError, InvalidOrder, InsufficientFunds, OrderNotFound, DDoSProtection } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bittrex extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bittrex',\n            'name': 'Bittrex',\n            'countries': 'US',\n            'version': 'v1.1',\n            'rateLimit': 1500,\n            'hasAlreadyAuthenticatedSuccessfully': false, // a workaround for APIKEY_INVALID\n            // new metainfo interface\n            'has': {\n                'CORS': true,\n                'fetchTickers': true,\n                'fetchOHLCV': true,\n                'fetchOrder': true,\n                'fetchOrders': true,\n                'fetchClosedOrders': 'emulated',\n                'fetchOpenOrders': true,\n                'fetchMyTrades': false,\n                'fetchCurrencies': true,\n                'withdraw': true,\n            },\n            'timeframes': {\n                '1m': 'oneMin',\n                '5m': 'fiveMin',\n                '30m': 'thirtyMin',\n                '1h': 'hour',\n                '1d': 'day',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766352-cf0b3c26-5ed5-11e7-82b7-f3826b7a97d8.jpg',\n                'api': {\n                    'public': 'https://bittrex.com/api',\n                    'account': 'https://bittrex.com/api',\n                    'market': 'https://bittrex.com/api',\n                    'v2': 'https://bittrex.com/api/v2.0/pub',\n                },\n                'www': 'https://bittrex.com',\n                'doc': [\n                    'https://bittrex.com/Home/Api',\n                    'https://www.npmjs.org/package/node.bittrex.api',\n                ],\n                'fees': [\n                    'https://bittrex.com/Fees',\n                    'https://support.bittrex.com/hc/en-us/articles/115000199651-What-fees-does-Bittrex-charge-',\n                ],\n            },\n            'api': {\n                'v2': {\n                    'get': [\n                        'currencies/GetBTCPrice',\n                        'market/GetTicks',\n                        'market/GetLatestTick',\n                        'Markets/GetMarketSummaries',\n                        'market/GetLatestTick',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'currencies',\n                        'markethistory',\n                        'markets',\n                        'marketsummaries',\n                        'marketsummary',\n                        'orderbook',\n                        'ticker',\n                    ],\n                },\n                'account': {\n                    'get': [\n                        'balance',\n                        'balances',\n                        'depositaddress',\n                        'deposithistory',\n                        'order',\n                        'orderhistory',\n                        'withdrawalhistory',\n                        'withdraw',\n                    ],\n                },\n                'market': {\n                    'get': [\n                        'buylimit',\n                        'buymarket',\n                        'cancel',\n                        'openorders',\n                        'selllimit',\n                        'sellmarket',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'maker': 0.0025,\n                    'taker': 0.0025,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.001,\n                        'LTC': 0.01,\n                        'DOGE': 2,\n                        'VTC': 0.02,\n                        'PPC': 0.02,\n                        'FTC': 0.2,\n                        'RDD': 2,\n                        'NXT': 2,\n                        'DASH': 0.002,\n                        'POT': 0.002,\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                        'LTC': 0,\n                        'DOGE': 0,\n                        'VTC': 0,\n                        'PPC': 0,\n                        'FTC': 0,\n                        'RDD': 0,\n                        'NXT': 0,\n                        'DASH': 0,\n                        'POT': 0,\n                    },\n                },\n            },\n        });\n    }\n\n    costToPrecision (symbol, cost) {\n        return this.truncate (parseFloat (cost), this.markets[symbol]['precision']['price']);\n    }\n\n    feeToPrecision (symbol, fee) {\n        return this.truncate (parseFloat (fee), this.markets[symbol]['precision']['price']);\n    }\n\n    async fetchMarkets () {\n        let response = await this.v2GetMarketsGetMarketSummaries ();\n        let result = [];\n        for (let i = 0; i < response['result'].length; i++) {\n            let market = response['result'][i]['Market'];\n            let id = market['MarketName'];\n            let base = market['MarketCurrency'];\n            let quote = market['BaseCurrency'];\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            let precision = {\n                'amount': 8,\n                'price': 8,\n            };\n            let active = market['IsActive'];\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': active,\n                'info': market,\n                'lot': Math.pow (10, -precision['amount']),\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': market['MinTradeSize'],\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                },\n            }));\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.accountGetBalances ();\n        let balances = response['result'];\n        let result = { 'info': balances };\n        let indexed = this.indexBy (balances, 'Currency');\n        let keys = Object.keys (indexed);\n        for (let i = 0; i < keys.length; i++) {\n            let id = keys[i];\n            let currency = this.commonCurrencyCode (id);\n            let account = this.account ();\n            let balance = indexed[id];\n            let free = parseFloat (balance['Available']);\n            let total = parseFloat (balance['Balance']);\n            let used = total - free;\n            account['free'] = free;\n            account['used'] = used;\n            account['total'] = total;\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetOrderbook (this.extend ({\n            'market': this.marketId (symbol),\n            'type': 'both',\n        }, params));\n        let orderbook = response['result'];\n        if ('type' in params) {\n            if (params['type'] === 'buy') {\n                orderbook = {\n                    'buy': response['result'],\n                    'sell': [],\n                };\n            } else if (params['type'] === 'sell') {\n                orderbook = {\n                    'buy': [],\n                    'sell': response['result'],\n                };\n            }\n        }\n        return this.parseOrderBook (orderbook, undefined, 'buy', 'sell', 'Rate', 'Quantity');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.parse8601 (ticker['TimeStamp']);\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'High'),\n            'low': this.safeFloat (ticker, 'Low'),\n            'bid': this.safeFloat (ticker, 'Bid'),\n            'ask': this.safeFloat (ticker, 'Ask'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'Last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'Volume'),\n            'quoteVolume': this.safeFloat (ticker, 'BaseVolume'),\n            'info': ticker,\n        };\n    }\n\n    async fetchCurrencies (params = {}) {\n        let response = await this.publicGetCurrencies (params);\n        let currencies = response['result'];\n        let result = {};\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let id = currency['Currency'];\n            // todo: will need to rethink the fees\n            // to add support for multiple withdrawal/deposit methods and\n            // differentiated fees for each particular method\n            let code = this.commonCurrencyCode (id);\n            let precision = 8; // default precision, todo: fix \"magic constants\"\n            result[code] = {\n                'id': id,\n                'code': code,\n                'info': currency,\n                'name': currency['CurrencyLong'],\n                'active': currency['IsActive'],\n                'status': 'ok',\n                'fee': currency['TxFee'], // todo: redesign\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': currency['TxFee'],\n                        'max': Math.pow (10, precision),\n                    },\n                },\n            };\n        }\n        return result;\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetMarketsummaries (params);\n        let tickers = response['result'];\n        let result = {};\n        for (let t = 0; t < tickers.length; t++) {\n            let ticker = tickers[t];\n            let id = ticker['MarketName'];\n            let market = undefined;\n            let symbol = id;\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            } else {\n                symbol = this.parseSymbol (id);\n            }\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetMarketsummary (this.extend ({\n            'market': market['id'],\n        }, params));\n        let ticker = response['result'][0];\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = this.parse8601 (trade['TimeStamp']);\n        let side = undefined;\n        if (trade['OrderType'] === 'BUY') {\n            side = 'buy';\n        } else if (trade['OrderType'] === 'SELL') {\n            side = 'sell';\n        }\n        let id = undefined;\n        if ('Id' in trade)\n            id = trade['Id'].toString ();\n        return {\n            'id': id,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': 'limit',\n            'side': side,\n            'price': parseFloat (trade['Price']),\n            'amount': parseFloat (trade['Quantity']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetMarkethistory (this.extend ({\n            'market': market['id'],\n        }, params));\n        if ('result' in response) {\n            if (typeof response['result'] !== 'undefined')\n                return this.parseTrades (response['result'], market, since, limit);\n        }\n        throw new ExchangeError (this.id + ' fetchTrades() returned undefined response');\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1d', since = undefined, limit = undefined) {\n        let timestamp = this.parse8601 (ohlcv['T']);\n        return [\n            timestamp,\n            ohlcv['O'],\n            ohlcv['H'],\n            ohlcv['L'],\n            ohlcv['C'],\n            ohlcv['V'],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'tickInterval': this.timeframes[timeframe],\n            'marketName': market['id'],\n        };\n        let response = await this.v2GetMarketGetTicks (this.extend (request, params));\n        if ('result' in response) {\n            if (response['result'])\n                return this.parseOHLCVs (response['result'], market, timeframe, since, limit);\n        }\n        throw new ExchangeError (this.id + ' returned an empty or unrecognized response: ' + this.json (response));\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['market'] = market['id'];\n        }\n        let response = await this.marketGetOpenorders (this.extend (request, params));\n        let orders = this.parseOrders (response['result'], market, since, limit);\n        return this.filterOrdersBySymbol (orders, symbol);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type !== 'limit')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'marketGet' + this.capitalize (side) + type;\n        let order = {\n            'market': market['id'],\n            'quantity': this.amountToPrecision (symbol, amount),\n            'rate': this.priceToPrecision (symbol, price),\n        };\n        // if (type == 'limit')\n        //     order['rate'] = this.priceToPrecision (symbol, price);\n        let response = await this[method] (this.extend (order, params));\n        let orderIdField = this.getOrderIdField ();\n        let result = {\n            'info': response,\n            'id': response['result'][orderIdField],\n        };\n        return result;\n    }\n\n    getOrderIdField () {\n        return 'uuid';\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = undefined;\n        try {\n            let orderIdField = this.getOrderIdField ();\n            let request = {};\n            request[orderIdField] = id;\n            response = await this.marketGetCancel (this.extend (request, params));\n        } catch (e) {\n            if (this.last_json_response) {\n                let message = this.safeString (this.last_json_response, 'message');\n                if (message === 'ORDER_NOT_OPEN')\n                    throw new InvalidOrder (this.id + ' cancelOrder() error: ' + this.last_http_response);\n                if (message === 'UUID_INVALID')\n                    throw new OrderNotFound (this.id + ' cancelOrder() error: ' + this.last_http_response);\n            }\n            throw e;\n        }\n        return response;\n    }\n\n    parseSymbol (id) {\n        let [ quote, base ] = id.split ('-');\n        base = this.commonCurrencyCode (base);\n        quote = this.commonCurrencyCode (quote);\n        return base + '/' + quote;\n    }\n\n    parseOrder (order, market = undefined) {\n        let side = this.safeString (order, 'OrderType');\n        if (typeof side === 'undefined')\n            side = this.safeString (order, 'Type');\n        let isBuyOrder = (side === 'LIMIT_BUY') || (side === 'BUY');\n        side = isBuyOrder ? 'buy' : 'sell';\n        let status = 'open';\n        if (('Closed' in order) && order['Closed']) {\n            status = 'closed';\n        } else if (('CancelInitiated' in order) && order['CancelInitiated']) {\n            status = 'canceled';\n        }\n        let symbol = undefined;\n        if (!market) {\n            if ('Exchange' in order) {\n                let marketId = order['Exchange'];\n                if (marketId in this.markets_by_id)\n                    market = this.markets_by_id[marketId];\n                else\n                    symbol = this.parseSymbol (marketId);\n            }\n        }\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = undefined;\n        if ('Opened' in order)\n            timestamp = this.parse8601 (order['Opened']);\n        if ('TimeStamp' in order)\n            timestamp = this.parse8601 (order['TimeStamp']);\n        if ('Created' in order)\n            timestamp = this.parse8601 (order['Created']);\n        let fee = undefined;\n        let commission = undefined;\n        if ('Commission' in order) {\n            commission = 'Commission';\n        } else if ('CommissionPaid' in order) {\n            commission = 'CommissionPaid';\n        }\n        if (commission) {\n            fee = {\n                'cost': parseFloat (order[commission]),\n            };\n            if (market)\n                fee['currency'] = market['quote'];\n        }\n        let price = this.safeFloat (order, 'Limit');\n        let cost = this.safeFloat (order, 'Price');\n        let amount = this.safeFloat (order, 'Quantity');\n        let remaining = this.safeFloat (order, 'QuantityRemaining', 0.0);\n        let filled = amount - remaining;\n        if (!cost) {\n            if (price && amount)\n                cost = price * amount;\n        }\n        if (!price) {\n            if (cost && filled)\n                price = cost / filled;\n        }\n        let average = this.safeFloat (order, 'PricePerUnit');\n        let id = this.safeString (order, 'OrderUuid');\n        if (typeof id === 'undefined')\n            id = this.safeString (order, 'OrderId');\n        let result = {\n            'info': order,\n            'id': id,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': 'limit',\n            'side': side,\n            'price': price,\n            'cost': cost,\n            'average': average,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': fee,\n        };\n        return result;\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = undefined;\n        try {\n            let orderIdField = this.getOrderIdField ();\n            let request = {};\n            request[orderIdField] = id;\n            response = await this.accountGetOrder (this.extend (request, params));\n        } catch (e) {\n            if (this.last_json_response) {\n                let message = this.safeString (this.last_json_response, 'message');\n                if (message === 'UUID_INVALID')\n                    throw new OrderNotFound (this.id + ' fetchOrder() error: ' + this.last_http_response);\n            }\n            throw e;\n        }\n        return this.parseOrder (response['result']);\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['market'] = market['id'];\n        }\n        let response = await this.accountGetOrderhistory (this.extend (request, params));\n        let orders = this.parseOrders (response['result'], market, since, limit);\n        if (symbol)\n            return this.filterOrdersBySymbol (orders, symbol);\n        return orders;\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, since, limit, params);\n        return this.filterBy (orders, 'status', 'closed');\n    }\n\n    currencyId (currency) {\n        if (currency === 'BCH')\n            return 'BCC';\n        return currency;\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        let currencyId = this.currencyId (currency);\n        let response = await this.accountGetDepositaddress (this.extend ({\n            'currency': currencyId,\n        }, params));\n        let address = this.safeString (response['result'], 'Address');\n        let message = this.safeString (response, 'message');\n        let status = 'ok';\n        if (!address || message === 'ADDRESS_GENERATING')\n            status = 'pending';\n        return {\n            'currency': currency,\n            'address': address,\n            'status': status,\n            'info': response,\n        };\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        let currencyId = this.currencyId (currency);\n        let request = {\n            'currency': currencyId,\n            'quantity': amount,\n            'address': address,\n        };\n        if (tag)\n            request['paymentid'] = tag;\n        let response = await this.accountGetWithdraw (this.extend (request, params));\n        let id = undefined;\n        if ('result' in response) {\n            if ('uuid' in response['result'])\n                id = response['result']['uuid'];\n        }\n        return {\n            'info': response,\n            'id': id,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api] + '/';\n        if (api !== 'v2')\n            url += this.version + '/';\n        if (api === 'public') {\n            url += api + '/' + method.toLowerCase () + path;\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else if (api === 'v2') {\n            url += path;\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            url += api + '/';\n            if (((api === 'account') && (path !== 'withdraw')) || (path === 'openorders'))\n                url += method.toLowerCase ();\n            url += path + '?' + this.urlencode (this.extend ({\n                'nonce': nonce,\n                'apikey': this.apiKey,\n            }, params));\n            let signature = this.hmac (this.encode (url), this.encode (this.secret), 'sha512');\n            headers = { 'apisign': signature };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    throwExceptionOnError (response) {\n        if ('message' in response) {\n            if (response['message'] === 'APISIGN_NOT_PROVIDED')\n                throw new AuthenticationError (this.id + ' ' + this.json (response));\n            if (response['message'] === 'INVALID_SIGNATURE')\n                throw new AuthenticationError (this.id + ' ' + this.json (response));\n            if (response['message'] === 'INSUFFICIENT_FUNDS')\n                throw new InsufficientFunds (this.id + ' ' + this.json (response));\n            if (response['message'] === 'MIN_TRADE_REQUIREMENT_NOT_MET')\n                throw new InvalidOrder (this.id + ' ' + this.json (response));\n            if (response['message'] === 'APIKEY_INVALID') {\n                if (this.hasAlreadyAuthenticatedSuccessfully) {\n                    throw new DDoSProtection (this.id + ' ' + this.json (response));\n                } else {\n                    throw new AuthenticationError (this.id + ' ' + this.json (response));\n                }\n            }\n            if (response['message'] === 'DUST_TRADE_DISALLOWED_MIN_VALUE_50K_SAT')\n                throw new InvalidOrder (this.id + ' order cost should be over 50k satoshi ' + this.json (response));\n        }\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code >= 400) {\n            if (body[0] === '{') {\n                let response = JSON.parse (body);\n                this.throwExceptionOnError (response);\n                if ('success' in response) {\n                    let success = response['success'];\n                    if (typeof success === 'string')\n                        success = (success === 'true') ? true : false;\n                    if (!success) {\n                        this.throwExceptionOnError (response);\n                        throw new ExchangeError (this.id + ' ' + this.json (response));\n                    }\n                }\n            }\n        }\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('success' in response) {\n            let success = response['success'];\n            if (typeof success === 'string')\n                success = (success === 'true') ? true : false;\n            if (success) {\n                // a workaround for APIKEY_INVALID\n                if ((api === 'account') || (api === 'market'))\n                    this.hasAlreadyAuthenticatedSuccessfully = true;\n                return response;\n            }\n        }\n        this.throwExceptionOnError (response);\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class bl3p extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bl3p',\n            'name': 'BL3P',\n            'countries': [ 'NL', 'EU' ], // Netherlands, EU\n            'rateLimit': 1000,\n            'version': '1',\n            'comment': 'An exchange market by BitonicNL',\n            'has': {\n                'CORS': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28501752-60c21b82-6feb-11e7-818b-055ee6d0e754.jpg',\n                'api': 'https://api.bl3p.eu',\n                'www': [\n                    'https://bl3p.eu',\n                    'https://bitonic.nl',\n                ],\n                'doc': [\n                    'https://github.com/BitonicNL/bl3p-api/tree/master/docs',\n                    'https://bl3p.eu/api',\n                    'https://bitonic.nl/en/api',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        '{market}/ticker',\n                        '{market}/orderbook',\n                        '{market}/trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        '{market}/money/depth/full',\n                        '{market}/money/order/add',\n                        '{market}/money/order/cancel',\n                        '{market}/money/order/result',\n                        '{market}/money/orders',\n                        '{market}/money/orders/history',\n                        '{market}/money/trades/fetch',\n                        'GENMKT/money/info',\n                        'GENMKT/money/deposit_address',\n                        'GENMKT/money/new_deposit_address',\n                        'GENMKT/money/wallet/history',\n                        'GENMKT/money/withdraw',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/EUR': { 'id': 'BTCEUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR', 'maker': 0.0025, 'taker': 0.0025 },\n                'LTC/EUR': { 'id': 'LTCEUR', 'symbol': 'LTC/EUR', 'base': 'LTC', 'quote': 'EUR', 'maker': 0.0025, 'taker': 0.0025 },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostGENMKTMoneyInfo ();\n        let data = response['data'];\n        let balance = data['wallets'];\n        let result = { 'info': data };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            if (currency in balance) {\n                if ('available' in balance[currency]) {\n                    account['free'] = parseFloat (balance[currency]['available']['value']);\n                }\n            }\n            if (currency in balance) {\n                if ('balance' in balance[currency]) {\n                    account['total'] = parseFloat (balance[currency]['balance']['value']);\n                }\n            }\n            if (account['total']) {\n                if (account['free']) {\n                    account['used'] = account['total'] - account['free'];\n                }\n            }\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    parseBidAsk (bidask, priceKey = 0, amountKey = 0) {\n        return [\n            bidask['price_int'] / 100000.0,\n            bidask['amount_int'] / 100000000.0,\n        ];\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetMarketOrderbook (this.extend ({\n            'market': market['id'],\n        }, params));\n        let orderbook = response['data'];\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetMarketTicker (this.extend ({\n            'market': this.marketId (symbol),\n        }, params));\n        let timestamp = ticker['timestamp'] * 1000;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume']['24h']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        return {\n            'id': trade['trade_id'].toString (),\n            'timestamp': trade['date'],\n            'datetime': this.iso8601 (trade['date']),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': trade['price_int'] / 100000.0,\n            'amount': trade['amount_int'] / 100000000.0,\n            'info': trade,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetMarketTrades (this.extend ({\n            'market': market['id'],\n        }, params));\n        let result = this.parseTrades (response['data']['trades'], market, since, limit);\n        return result;\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let market = this.market (symbol);\n        let order = {\n            'market': market['id'],\n            'amount_int': parseInt (amount * 100000000),\n            'fee_currency': market['quote'],\n            'type': (side == 'buy') ? 'bid' : 'ask',\n        };\n        if (type == 'limit')\n            order['price_int'] = parseInt (price * 100000.0);\n        let response = await this.privatePostMarketMoneyOrderAdd (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['data']['order_id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostMarketMoneyOrderCancel ({ 'order_id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let request = this.implodeParams (path, params);\n        let url = this.urls['api'] + '/' + this.version + '/' + request;\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({ 'nonce': nonce }, query));\n            let secret = this.base64ToBinary (this.secret);\n            let auth = request + \"\\0\" + body;\n            let signature = this.hmac (this.encode (auth), secret, 'sha512', 'base64');\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Rest-Key': this.apiKey,\n                'Rest-Sign': this.decode (signature),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst bittrex = require ('./bittrex.js');\nconst { AuthenticationError, InvalidOrder, InsufficientFunds, DDoSProtection } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class bleutrade extends bittrex {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bleutrade',\n            'name': 'Bleutrade',\n            'countries': 'BR', // Brazil\n            'rateLimit': 1000,\n            'version': 'v2',\n            'has': {\n                'CORS': true,\n                'fetchTickers': true,\n                'fetchOHLCV': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30303000-b602dbe6-976d-11e7-956d-36c5049c01e7.jpg',\n                'api': {\n                    'public': 'https://bleutrade.com/api',\n                    'account': 'https://bleutrade.com/api',\n                    'market': 'https://bleutrade.com/api',\n                },\n                'www': 'https://bleutrade.com',\n                'doc': 'https://bleutrade.com/help/API',\n                'fees': 'https://bleutrade.com/help/fees_and_deadlines',\n            },\n            'fees': {\n                'funding': {\n                    'ADC': 0.1,\n                    'BTA': 0.1,\n                    'BITB': 0.1,\n                    'BTC': 0.001,\n                    'BCH': 0.001,\n                    'BTCD': 0.001,\n                    'BTG': 0.001,\n                    'BLK': 0.1,\n                    'CDN': 0.1,\n                    'CLAM': 0.01,\n                    'DASH': 0.001,\n                    'DCR': 0.05,\n                    'DGC': 0.1,\n                    'DP': 0.1,\n                    'DPC': 0.1,\n                    'DOGE': 0.0,\n                    'EFL': 0.1,\n                    'ETH': 0.01,\n                    'EXP': 0.1,\n                    'FJC': 0.1,\n                    'BSTY': 0.001,\n                    'GB': 0.1,\n                    'NLG': 0.1,\n                    'HTML': 1.0,\n                    'LTC': 0.001,\n                    'MONA': 0.01,\n                    'MOON': 1.0,\n                    'NMC': 0.015,\n                    'NEOS': 0.1,\n                    'NVC': 0.05,\n                    'OK': 0.1,\n                    'PPC': 0.1,\n                    'POT': 0.1,\n                    'XPM': 0.001,\n                    'QTUM': 0.1,\n                    'RDD': 0.1,\n                    'SLR': 0.1,\n                    'START': 0.1,\n                    'SLG': 0.1,\n                    'TROLL': 0.1,\n                    'UNO': 0.01,\n                    'VRC': 0.1,\n                    'VTC': 0.1,\n                    'XVP': 0.1,\n                    'WDC': 0.001,\n                    'ZET': 0.1,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetMarkets ();\n        let result = [];\n        for (let p = 0; p < markets['result'].length; p++) {\n            let market = markets['result'][p];\n            let id = market['MarketName'];\n            let base = market['MarketCurrency'];\n            let quote = market['BaseCurrency'];\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            let precision = {\n                'amount': 8,\n                'price': 8,\n            };\n            let active = market['IsActive'];\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': active,\n                'info': market,\n                'lot': Math.pow (10, -precision['amount']),\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': market['MinTradeSize'],\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': 0,\n                        'max': undefined,\n                    },\n                },\n            }));\n        }\n        return result;\n    }\n\n    getOrderIdField () {\n        return 'orderid';\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetOrderbook (this.extend ({\n            'market': this.marketId (symbol),\n            'type': 'ALL',\n            'depth': 50,\n        }, params));\n        let orderbook = response['result'];\n        return this.parseOrderBook (orderbook, undefined, 'buy', 'sell', 'Rate', 'Quantity');\n    }\n\n    throwExceptionOnError (response) {\n        if ('message' in response) {\n            if (response['message'] === 'Insufficient funds!')\n                throw new InsufficientFunds (this.id + ' ' + this.json (response));\n            if (response['message'] === 'MIN_TRADE_REQUIREMENT_NOT_MET')\n                throw new InvalidOrder (this.id + ' ' + this.json (response));\n            if (response['message'] === 'APIKEY_INVALID') {\n                if (this.hasAlreadyAuthenticatedSuccessfully) {\n                    throw new DDoSProtection (this.id + ' ' + this.json (response));\n                } else {\n                    throw new AuthenticationError (this.id + ' ' + this.json (response));\n                }\n            }\n            if (response['message'] === 'DUST_TRADE_DISALLOWED_MIN_VALUE_50K_SAT')\n                throw new InvalidOrder (this.id + ' order cost should be over 50k satoshi ' + this.json (response));\n        }\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InvalidOrder, AuthenticationError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class braziliex extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'braziliex',\n            'name': 'Braziliex',\n            'countries': 'BR',\n            'rateLimit': 1000,\n            'has': {\n                'fetchTickers': true,\n                'fetchOpenOrders': true,\n                'fetchMyTrades': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/34703593-c4498674-f504-11e7-8d14-ff8e44fb78c1.jpg',\n                'api': 'https://braziliex.com/api/v1',\n                'www': 'https://braziliex.com/',\n                'doc': 'https://braziliex.com/exchange/api.php',\n                'fees': 'https://braziliex.com/exchange/fees.php',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'currencies',\n                        'ticker',\n                        'ticker/{market}',\n                        'orderbook/{market}',\n                        'tradehistory/{market}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balance',\n                        'complete_balance',\n                        'open_orders',\n                        'trade_history',\n                        'deposit_address',\n                        'sell',\n                        'buy',\n                        'cancel_order',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.005,\n                    'taker': 0.005,\n                },\n            },\n            'precision': {\n                'amount': 8,\n                'price': 8,\n            },\n        });\n    }\n\n    async fetchCurrencies (params = {}) {\n        let currencies = await this.publicGetCurrencies (params);\n        let ids = Object.keys (currencies);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let currency = currencies[id];\n            let precision = currency['decimal'];\n            let uppercase = id.toUpperCase ();\n            let code = this.commonCurrencyCode (uppercase);\n            let active = currency['active'] == 1;\n            let status = 'ok';\n            if (currency['under_maintenance'] != 0) {\n                active = false;\n                status = 'maintenance';\n            }\n            let canWithdraw = currency['is_withdrawal_active'] == 1;\n            let canDeposit = currency['is_deposit_active'] == 1;\n            if (!canWithdraw || !canDeposit)\n                active = false;\n            result[code] = {\n                'id': id,\n                'code': code,\n                'name': currency['name'],\n                'active': active,\n                'status': status,\n                'precision': precision,\n                'wallet': {\n                    'address': undefined,\n                    'extra': undefined,\n                    'withdraw': {\n                        'active': canWithdraw,\n                        'fee': currency['txWithdrawalFee'],\n                    },\n                    'deposit': {\n                        'active': canDeposit,\n                        'fee': currency['txDepositFee'],\n                    },\n                },\n                'limits': {\n                    'amount': {\n                        'min': currency['minAmountTrade'],\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': currency['MinWithdrawal'],\n                        'max': Math.pow (10, precision),\n                    },\n                    'deposit': {\n                        'min': currency['minDeposit'],\n                        'max': undefined,\n                    },\n                },\n                'info': currency,\n            };\n        }\n        return result;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetTicker ();\n        let ids = Object.keys (markets);\n        let result = [];\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = markets[id];\n            let idUpperCase = id.toUpperCase ();\n            let [ base, quote ] = idUpperCase.split ('_');\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            let active = market['active'] == 1;\n            let precision = {\n                'amount': 8,\n                'price': 8,\n            };\n            let lot = Math.pow (10, -precision['amount']);\n            result.push ({\n                'id': id,\n                'symbol': symbol.toUpperCase (),\n                'base': base,\n                'quote': quote,\n                'active': active,\n                'lot': lot,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': Math.pow (10, precision['amount']),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision['price']),\n                        'max': Math.pow (10, precision['price']),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                },\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let symbol = market['symbol'];\n        let timestamp = ticker['date'];\n        ticker = ticker['ticker'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['highestBid24']),\n            'low': parseFloat (ticker['lowestAsk24']),\n            'bid': parseFloat (ticker['highestBid']),\n            'ask': parseFloat (ticker['lowestAsk']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': parseFloat (ticker['percentChange']),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['baseVolume24']),\n            'quoteVolume': parseFloat (ticker['quoteVolume24']),\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetTickerMarket (this.extend ({\n            'market': market['id'],\n        }, params));\n        ticker = {\n            'date': this.milliseconds (),\n            'ticker': ticker,\n        };\n        return this.parseTicker (ticker, market);\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTicker (params);\n        let result = {};\n        let timestamp = this.milliseconds ();\n        let ids = Object.keys (tickers);\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = {\n                'date': timestamp,\n                'ticker': tickers[id],\n            };\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderbookMarket (this.extend ({\n            'market': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'amount');\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = undefined;\n        if ('date_exec' in trade) {\n            timestamp = this.parse8601 (trade['date_exec']);\n        } else {\n            timestamp = this.parse8601 (trade['date']);\n        }\n        let price = parseFloat (trade['price']);\n        let amount = parseFloat (trade['amount']);\n        let symbol = market['symbol'];\n        let cost = parseFloat (trade['total']);\n        let orderId = this.safeString (trade, 'order_number');\n        return {\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'id': this.safeString (trade, '_id'),\n            'order': orderId,\n            'type': 'limit',\n            'side': trade['type'],\n            'price': price,\n            'amount': amount,\n            'cost': cost,\n            'fee': undefined,\n            'info': trade,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let trades = await this.publicGetTradehistoryMarket (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseTrades (trades, market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privatePostCompleteBalance (params);\n        let result = { 'info': balances };\n        let currencies = Object.keys (balances);\n        for (let i = 0; i < currencies.length; i++) {\n            let id = currencies[i];\n            let balance = balances[id];\n            let currency = this.commonCurrencyCode (id);\n            let account = {\n                'free': parseFloat (balance['available']),\n                'used': 0.0,\n                'total': parseFloat (balance['total']),\n            };\n            account['used'] = account['total'] - account['free'];\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    parseOrder (order, market = undefined) {\n        let symbol = undefined;\n        if (!market) {\n            let marketId = this.safeString (order, 'market');\n            if (marketId)\n                if (marketId in this.markets_by_id)\n                    market = this.markets_by_id[marketId];\n        }\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = this.safeValue (order, 'timestamp');\n        if (!timestamp)\n            timestamp = this.parse8601 (order['date']);\n        let price = parseFloat (order['price']);\n        let cost = this.safeFloat (order, 'total', 0.0);\n        let amount = this.safeFloat (order, 'amount');\n        let filledPercentage = this.safeFloat (order, 'progress');\n        let filled = amount * filledPercentage;\n        let remaining = this.amountToPrecision (symbol, amount - filled);\n        let info = order;\n        if ('info' in info)\n            info = order['info'];\n        return {\n            'id': order['order_number'],\n            'datetime': this.iso8601 (timestamp),\n            'timestamp': timestamp,\n            'status': 'open',\n            'symbol': symbol,\n            'type': 'limit',\n            'side': order['type'],\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'trades': undefined,\n            'fee': this.safeValue (order, 'fee'),\n            'info': info,\n        };\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'privatePost' + this.capitalize (side);\n        let response = await this[method] (this.extend ({\n            'market': market['id'],\n            // 'price': this.priceToPrecision (symbol, price),\n            // 'amount': this.amountToPrecision (symbol, amount),\n            'price': price,\n            'amount': amount,\n        }, params));\n        let success = this.safeInteger (response, 'success');\n        if (success != 1)\n            throw new InvalidOrder (this.id + ' ' + this.json (response));\n        let parts = response['message'].split (' / ');\n        parts = parts.slice (1);\n        let feeParts = parts[5].split (' ');\n        let order = this.parseOrder ({\n            'timestamp': this.milliseconds (),\n            'order_number': response['order_number'],\n            'type': parts[0].toLowerCase (),\n            'market': parts[0].toLowerCase (),\n            'amount': parts[2].split (' ')[1],\n            'price': parts[3].split (' ')[1],\n            'total': parts[4].split (' ')[1],\n            'fee': {\n                'cost': parseFloat (feeParts[1]),\n                'currency': feeParts[2],\n            },\n            'progress': '0.0',\n            'info': response,\n        }, market);\n        let id = order['id'];\n        this.orders[id] = order;\n        return order;\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let result = await this.privatePostCancelOrder (this.extend ({\n            'order_number': id,\n            'market': market['id'],\n        }, params));\n        return result;\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orders = await this.privatePostOpenOrders (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseOrders (orders['order_open'], market, since, limit);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let trades = await this.privatePostTradeHistory (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseTrades (trades['trade_history'], market, since, limit);\n    }\n\n    async fetchDepositAddress (currencyCode, params = {}) {\n        await this.loadMarkets ();\n        let currency = this.currency (currencyCode);\n        let response = await this.privatePostDepositAddress (this.extend ({\n            'currency': currency['id'],\n        }, params));\n        let address = this.safeString (response['deposit_address'], 'address');\n        if (!address)\n            throw new ExchangeError (this.id + ' fetchDepositAddress failed: ' + this.last_http_response);\n        return {\n            'currency': currencyCode,\n            'address': address,\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + api;\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            url += '/' + this.implodeParams (path, params);\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            query = this.extend ({\n                'command': path,\n                'nonce': this.nonce (),\n            }, query);\n            body = this.urlencode (query);\n            let signature = this.hmac (this.encode (body), this.encode (this.secret), 'sha512');\n            headers = {\n                'Content-type': 'application/x-www-form-urlencoded',\n                'Key': this.apiKey,\n                'Sign': this.decode (signature),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('success' in response) {\n            let success = this.safeInteger (response, 'success');\n            if (success == 0) {\n                let message = this.safeString (response, 'message');\n                if (message == 'Invalid APIKey')\n                    throw new AuthenticationError (message);\n                throw new ExchangeError (message);\n            }\n        }\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class btcbox extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'btcbox',\n            'name': 'BtcBox',\n            'countries': 'JP',\n            'rateLimit': 1000,\n            'version': 'v1',\n            'has': {\n                'CORS': false,\n                'fetchOHLCV': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/31275803-4df755a8-aaa1-11e7-9abb-11ec2fad9f2d.jpg',\n                'api': 'https://www.btcbox.co.jp/api',\n                'www': 'https://www.btcbox.co.jp/',\n                'doc': 'https://www.btcbox.co.jp/help/asm',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'depth',\n                        'orders',\n                        'ticker',\n                        'allticker',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balance',\n                        'trade_add',\n                        'trade_cancel',\n                        'trade_list',\n                        'trade_view',\n                        'wallet',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/JPY': { 'id': 'BTC/JPY', 'symbol': 'BTC/JPY', 'base': 'BTC', 'quote': 'JPY' },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privatePostBalance ();\n        let result = { 'info': balances };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            if (lowercase == 'dash')\n                lowercase = 'drk';\n            let account = this.account ();\n            let free = lowercase + '_balance';\n            let used = lowercase + '_lock';\n            if (free in balances)\n                account['free'] = parseFloat (balances[free]);\n            if (used in balances)\n                account['used'] = parseFloat (balances[used]);\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {};\n        let numSymbols = this.symbols.length;\n        if (numSymbols > 1)\n            request['coin'] = market['id'];\n        let orderbook = await this.publicGetDepth (this.extend (request, params));\n        let result = this.parseOrderBook (orderbook);\n        result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'buy'),\n            'ask': this.safeFloat (ticker, 'sell'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'vol'),\n            'quoteVolume': this.safeFloat (ticker, 'volume'),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetAllticker (params);\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {};\n        let numSymbols = this.symbols.length;\n        if (numSymbols > 1)\n            request['coin'] = market['id'];\n        let ticker = await this.publicGetTicker (this.extend (request, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        return {\n            'info': trade,\n            'id': trade['tid'],\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['type'],\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {};\n        let numSymbols = this.symbols.length;\n        if (numSymbols > 1)\n            request['coin'] = market['id'];\n        let response = await this.publicGetOrders (this.extend (request, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'amount': amount,\n            'price': price,\n            'type': side,\n        };\n        let numSymbols = this.symbols.length;\n        if (numSymbols > 1)\n            request['coin'] = market['id'];\n        let response = await this.privatePostTradeAdd (this.extend (request, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostTradeCancel (this.extend ({\n            'id': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + path;\n        if (api == 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let query = this.extend ({\n                'key': this.apiKey,\n                'nonce': nonce,\n            }, params);\n            let request = this.urlencode (query);\n            let secret = this.hash (this.encode (this.secret));\n            query['signature'] = this.hmac (this.encode (request), this.encode (secret));\n            body = this.urlencode (query);\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('result' in response)\n            if (!response['result'])\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class btcchina extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'btcchina',\n            'name': 'BTCChina',\n            'countries': 'CN',\n            'rateLimit': 1500,\n            'version': 'v1',\n            'has': {\n                'CORS': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766368-465b3286-5ed6-11e7-9a11-0f6467e1d82b.jpg',\n                'api': {\n                    'plus': 'https://plus-api.btcchina.com/market',\n                    'public': 'https://data.btcchina.com/data',\n                    'private': 'https://api.btcchina.com/api_trade_v1.php',\n                },\n                'www': 'https://www.btcchina.com',\n                'doc': 'https://www.btcchina.com/apidocs'\n            },\n            'api': {\n                'plus': {\n                    'get': [\n                        'orderbook',\n                        'ticker',\n                        'trade',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'historydata',\n                        'orderbook',\n                        'ticker',\n                        'trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'BuyIcebergOrder',\n                        'BuyOrder',\n                        'BuyOrder2',\n                        'BuyStopOrder',\n                        'CancelIcebergOrder',\n                        'CancelOrder',\n                        'CancelStopOrder',\n                        'GetAccountInfo',\n                        'getArchivedOrder',\n                        'getArchivedOrders',\n                        'GetDeposits',\n                        'GetIcebergOrder',\n                        'GetIcebergOrders',\n                        'GetMarketDepth',\n                        'GetMarketDepth2',\n                        'GetOrder',\n                        'GetOrders',\n                        'GetStopOrder',\n                        'GetStopOrders',\n                        'GetTransactions',\n                        'GetWithdrawal',\n                        'GetWithdrawals',\n                        'RequestWithdrawal',\n                        'SellIcebergOrder',\n                        'SellOrder',\n                        'SellOrder2',\n                        'SellStopOrder',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/CNY': { 'id': 'btccny', 'symbol': 'BTC/CNY', 'base': 'BTC', 'quote': 'CNY', 'api': 'public', 'plus': false },\n                'LTC/CNY': { 'id': 'ltccny', 'symbol': 'LTC/CNY', 'base': 'LTC', 'quote': 'CNY', 'api': 'public', 'plus': false },\n                'LTC/BTC': { 'id': 'ltcbtc', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC', 'api': 'public', 'plus': false },\n                'BCH/CNY': { 'id': 'bcccny', 'symbol': 'BCH/CNY', 'base': 'BCH', 'quote': 'CNY', 'api': 'plus', 'plus': true },\n                'ETH/CNY': { 'id': 'ethcny', 'symbol': 'ETH/CNY', 'base': 'ETH', 'quote': 'CNY', 'api': 'plus', 'plus': true },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetTicker ({\n            'market': 'all',\n        });\n        let result = [];\n        let keys = Object.keys (markets);\n        for (let p = 0; p < keys.length; p++) {\n            let key = keys[p];\n            let market = markets[key];\n            let parts = key.split ('_');\n            let id = parts[1];\n            let base = id.slice (0, 3);\n            let quote = id.slice (3, 6);\n            base = base.toUpperCase ();\n            quote = quote.toUpperCase ();\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetAccountInfo ();\n        let balances = response['result'];\n        let result = { 'info': balances };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let account = this.account ();\n            if (lowercase in balances['balance'])\n                account['total'] = parseFloat (balances['balance'][lowercase]['amount']);\n            if (lowercase in balances['frozen'])\n                account['used'] = parseFloat (balances['frozen'][lowercase]['amount']);\n            account['free'] = account['total'] - account['used'];\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    createMarketRequest (market) {\n        let request = {};\n        let field = (market['plus']) ? 'symbol' : 'market';\n        request[field] = market['id'];\n        return request;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = market['api'] + 'GetOrderbook';\n        let request = this.createMarketRequest (market);\n        let orderbook = await this[method] (this.extend (request, params));\n        let timestamp = orderbook['date'] * 1000;\n        let result = this.parseOrderBook (orderbook, timestamp);\n        result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    parseTicker (ticker, market) {\n        let timestamp = ticker['date'] * 1000;\n        return {\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy']),\n            'ask': parseFloat (ticker['sell']),\n            'vwap': parseFloat (ticker['vwap']),\n            'open': parseFloat (ticker['open']),\n            'close': parseFloat (ticker['prev_close']),\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['vol']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTickerPlus (ticker, market) {\n        let timestamp = ticker['Timestamp'];\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['High']),\n            'low': parseFloat (ticker['Low']),\n            'bid': parseFloat (ticker['BidPrice']),\n            'ask': parseFloat (ticker['AskPrice']),\n            'vwap': undefined,\n            'open': parseFloat (ticker['Open']),\n            'close': parseFloat (ticker['PrevCls']),\n            'first': undefined,\n            'last': parseFloat (ticker['Last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['Volume24H']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = market['api'] + 'GetTicker';\n        let request = this.createMarketRequest (market);\n        let tickers = await this[method] (this.extend (request, params));\n        let ticker = tickers['ticker'];\n        if (market['plus'])\n            return this.parseTickerPlus (ticker, market);\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        return {\n            'id': trade['tid'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    parseTradePlus (trade, market) {\n        let timestamp = this.parse8601 (trade['timestamp']);\n        return {\n            'id': undefined,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['side'].toLowerCase (),\n            'price': trade['price'],\n            'amount': trade['size'],\n        };\n    }\n\n    parseTradesPlus (trades, market = undefined) {\n        let result = [];\n        for (let i = 0; i < trades.length; i++) {\n            result.push (this.parseTradePlus (trades[i], market));\n        }\n        return result;\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = market['api'] + 'GetTrade';\n        let request = this.createMarketRequest (market);\n        if (market['plus']) {\n            let now = this.milliseconds ();\n            request['start_time'] = now - 86400 * 1000;\n            request['end_time'] = now;\n        } else {\n            method += 's'; // trades vs trade\n        }\n        let response = await this[method] (this.extend (request, params));\n        if (market['plus']) {\n            return this.parseTradesPlus (response['trades'], market);\n        }\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'privatePost' + this.capitalize (side) + 'Order2';\n        let order = {};\n        let id = market['id'].toUpperCase ();\n        if (type == 'market') {\n            order['params'] = [ undefined, amount, id ];\n        } else {\n            order['params'] = [ price, amount, id ];\n        }\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = params['market']; // TODO fixme\n        return await this.privatePostCancelOrder (this.extend ({\n            'params': [ id, market ],\n        }, params));\n    }\n\n    nonce () {\n        return this.microseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api] + '/' + path;\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            let p = [];\n            if ('params' in params)\n                p = params['params'];\n            let nonce = this.nonce ();\n            let request = {\n                'method': path,\n                'id': nonce,\n                'params': p,\n            };\n            p = p.join (',');\n            body = this.json (request);\n            let query = (\n                'tonce=' + nonce +\n                '&accesskey=' + this.apiKey +\n                '&requestmethod=' + method.toLowerCase () +\n                '&id=' + nonce +\n                '&method=' + path +\n                '&params=' + p\n            );\n            let signature = this.hmac (this.encode (query), this.encode (this.secret), 'sha1');\n            let auth = this.encode (this.apiKey + ':' + signature);\n            headers = {\n                'Authorization': 'Basic ' + this.stringToBase64 (auth),\n                'Json-Rpc-Tonce': nonce,\n            };\n        } else {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst btcturk = require ('./btcturk.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class btcexchange extends btcturk {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'btcexchange',\n            'name': 'BTCExchange',\n            'countries': 'PH', // Philippines\n            'rateLimit': 1500,\n            'has': {\n                'CORS': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27993052-4c92911a-64aa-11e7-96d8-ec6ac3435757.jpg',\n                'api': 'https://www.btcexchange.ph/api',\n                'www': 'https://www.btcexchange.ph',\n                'doc': 'https://github.com/BTCTrader/broker-api-docs',\n            },\n            'markets': {\n                'BTC/PHP': { 'id': 'BTC/PHP', 'symbol': 'BTC/PHP', 'base': 'BTC', 'quote': 'PHP' },\n            },\n        });\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, OrderNotFound, NotSupported } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class btcmarkets extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'btcmarkets',\n            'name': 'BTC Markets',\n            'countries': 'AU', // Australia\n            'rateLimit': 1000, // market data cached for 1 second (trades cached for 2 seconds)\n            'has': {\n                'CORS': false,\n                'fetchOrder': true,\n                'fetchOrders': true,\n                'fetchClosedOrders': 'emulated',\n                'fetchOpenOrders': true,\n                'fetchMyTrades': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/29142911-0e1acfc2-7d5c-11e7-98c4-07d9532b29d7.jpg',\n                'api': 'https://api.btcmarkets.net',\n                'www': 'https://btcmarkets.net/',\n                'doc': 'https://github.com/BTCMarkets/API',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'market/{id}/tick',\n                        'market/{id}/orderbook',\n                        'market/{id}/trades',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'account/balance',\n                        'account/{id}/tradingfee',\n                    ],\n                    'post': [\n                        'fundtransfer/withdrawCrypto',\n                        'fundtransfer/withdrawEFT',\n                        'order/create',\n                        'order/cancel',\n                        'order/history',\n                        'order/open',\n                        'order/trade/history',\n                        'order/createBatch', // they promise it's coming soon...\n                        'order/detail',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/AUD': { 'id': 'BTC/AUD', 'symbol': 'BTC/AUD', 'base': 'BTC', 'quote': 'AUD', 'maker': 0.0085, 'taker': 0.0085 },\n                'LTC/AUD': { 'id': 'LTC/AUD', 'symbol': 'LTC/AUD', 'base': 'LTC', 'quote': 'AUD', 'maker': 0.0085, 'taker': 0.0085 },\n                'ETH/AUD': { 'id': 'ETH/AUD', 'symbol': 'ETH/AUD', 'base': 'ETH', 'quote': 'AUD', 'maker': 0.0085, 'taker': 0.0085 },\n                'ETC/AUD': { 'id': 'ETC/AUD', 'symbol': 'ETC/AUD', 'base': 'ETC', 'quote': 'AUD', 'maker': 0.0085, 'taker': 0.0085 },\n                'XRP/AUD': { 'id': 'XRP/AUD', 'symbol': 'XRP/AUD', 'base': 'XRP', 'quote': 'AUD', 'maker': 0.0085, 'taker': 0.0085 },\n                'BCH/AUD': { 'id': 'BCH/AUD', 'symbol': 'BCH/AUD', 'base': 'BCH', 'quote': 'AUD', 'maker': 0.0085, 'taker': 0.0085 },\n                'LTC/BTC': { 'id': 'LTC/BTC', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC', 'maker': 0.0022, 'taker': 0.0022 },\n                'ETH/BTC': { 'id': 'ETH/BTC', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC', 'maker': 0.0022, 'taker': 0.0022 },\n                'ETC/BTC': { 'id': 'ETC/BTC', 'symbol': 'ETC/BTC', 'base': 'ETC', 'quote': 'BTC', 'maker': 0.0022, 'taker': 0.0022 },\n                'XRP/BTC': { 'id': 'XRP/BTC', 'symbol': 'XRP/BTC', 'base': 'XRP', 'quote': 'BTC', 'maker': 0.0022, 'taker': 0.0022 },\n                'BCH/BTC': { 'id': 'BCH/BTC', 'symbol': 'BCH/BTC', 'base': 'BCH', 'quote': 'BTC', 'maker': 0.0022, 'taker': 0.0022 },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privateGetAccountBalance ();\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let multiplier = 100000000;\n            let total = parseFloat (balance['balance'] / multiplier);\n            let used = parseFloat (balance['pendingFunds'] / multiplier);\n            let free = total - used;\n            let account = {\n                'free': free,\n                'used': used,\n                'total': total,\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orderbook = await this.publicGetMarketIdOrderbook (this.extend ({\n            'id': market['id'],\n        }, params));\n        let timestamp = orderbook['timestamp'] * 1000;\n        return this.parseOrderBook (orderbook, timestamp);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['timestamp'] * 1000;\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['bestBid']),\n            'ask': parseFloat (ticker['bestAsk']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['lastPrice']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume24h']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetMarketIdTick (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['date'] * 1000;\n        return {\n            'info': trade,\n            'id': trade['tid'].toString (),\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetMarketIdTrades (this.extend ({\n            // 'since': 59868345231,\n            'id': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let multiplier = 100000000; // for price and volume\n        // does BTC Markets support market orders at all?\n        let orderSide = (side == 'buy') ? 'Bid' : 'Ask';\n        let order = this.ordered ({\n            'currency': market['quote'],\n        });\n        order['currency'] = market['quote'];\n        order['instrument'] = market['base'];\n        order['price'] = parseInt (price * multiplier);\n        order['volume'] = parseInt (amount * multiplier);\n        order['orderSide'] = orderSide;\n        order['ordertype'] = this.capitalize (type);\n        order['clientRequestId'] = this.nonce ().toString ();\n        let response = await this.privatePostOrderCreate (order);\n        return {\n            'info': response,\n            'id': response['id'].toString (),\n        };\n    }\n\n    async cancelOrders (ids) {\n        await this.loadMarkets ();\n        for (let i = 0; i < ids.length; i++) {\n            ids[i] = parseInt(ids[i]);\n        }\n        return await this.privatePostOrderCancel ({ 'orderIds': ids });\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.cancelOrders ([ id ]);\n    }\n\n    parseMyTrade (trade, market) {\n        let multiplier = 100000000;\n        let timestamp = trade['creationTime'];\n        let side = (trade['side'] == 'Bid') ? 'buy' : 'sell';\n        // BTCMarkets always charge in AUD for AUD-related transactions.\n        let currency = (market['quote'] == 'AUD') ? market['quote'] : market['base'];\n        return {\n            'info': trade,\n            'id': trade['id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': side,\n            'price': trade['price'] / multiplier,\n            'fee': {\n                'currency': currency,\n                'cost': trade['fee'] / multiplier,\n            },\n            'amount': trade['volume'] / multiplier,\n            'order': this.safeString (trade, 'orderId'),\n        };\n    }\n\n    parseMyTrades (trades, market = undefined, since = undefined, limit = undefined) {\n        let result = [];\n        for (let i = 0; i < trades.length; i++) {\n            let trade = this.parseMyTrade (trades[i], market);\n            result.push (trade);\n        }\n        return result;\n    }\n\n    parseOrder (order, market = undefined) {\n        let multiplier = 100000000;\n        let side = (order['orderSide'] == 'Bid') ? 'buy' : 'sell';\n        let type = (order['ordertype'] == 'Limit') ? 'limit' : 'market';\n        let timestamp = order['creationTime'];\n        if (!market) {\n            market = this.market(order['instrument'] + \"/\" + order['currency']);\n        }\n        let status = 'open';\n        if (order['status'] == 'Failed' || order['status'] == 'Cancelled' || order['status'] == 'Partially Cancelled' || order['status'] == 'Error') {\n            status = 'canceled';\n        } else if (order['status'] == \"Fully Matched\" || order['status'] == \"Partially Matched\") {\n            status = 'closed';\n        }\n        let price = this.safeFloat (order, 'price') / multiplier;\n        let amount = this.safeFloat (order, 'volume') / multiplier;\n        let remaining = this.safeFloat (order, 'openVolume', 0.0) / multiplier;\n        let filled = amount - remaining;\n        let cost = price * amount;\n        let trades = this.parseMyTrades (order['trades'], market);\n        let result = {\n            'info': order,\n            'id': order['id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': type,\n            'side': side,\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'trades': trades,\n            'fee': undefined,\n        };\n        return result;\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let ids = [ parseInt (id) ];\n        let response = await this.privatePostOrderDetail (this.extend ({\n            'orderIds': ids,\n        }, params));\n        let numOrders = response['orders'].length;\n        if (numOrders < 1)\n            throw new OrderNotFound (this.id + ' No matching order found: ' + id);\n        let order = response['orders'][0];\n        return this.parseOrder (order);\n    }\n\n    async prepareHistoryRequest (market, since = undefined, limit = undefined) {\n        let request = this.ordered ({\n            'currency': market['quote'],\n            'instrument': market['base'],\n        });\n        if (limit) {\n            request['limit'] = limit;\n        } else {\n            request['limit'] = 100;\n        }\n        if (since) {\n            request['since'] = since;\n        } else {\n            request['since'] = 0;\n        }\n        return request;\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new NotSupported (this.id + ': fetchOrders requires a `symbol` parameter.');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = this.prepareHistoryRequest (market, since, limit);\n        let response = await this.privatePostOrderHistory (this.extend (request, params));\n        return this.parseOrders (response['orders'], market);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new NotSupported (this.id + ': fetchOpenOrders requires a `symbol` parameter.');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = this.prepareHistoryRequest (market, since, limit);\n        let response = await this.privatePostOrderOpen (this.extend (request, params));\n        return this.parseOrders (response['orders'], market);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, since, limit, params);\n        return this.filterBy (orders, 'status', 'closed');\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new NotSupported (this.id + ': fetchMyTrades requires a `symbol` parameter.');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = this.prepareHistoryRequest (market, since, limit);\n        let response = await this.privatePostOrderTradeHistory (this.extend (request, params));\n        return this.parseMyTrades (response['trades'], market);\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let uri = '/' + this.implodeParams (path, params);\n        let url = this.urls['api'] + uri;\n        if (api == 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let auth = uri + \"\\n\" + nonce + \"\\n\";\n            headers = {\n                'Content-Type': 'application/json',\n                'apikey': this.apiKey,\n                'timestamp': nonce,\n            };\n            if (method == 'POST') {\n                body = this.json (params);\n                auth += body;\n            }\n            let secret = this.base64ToBinary (this.secret);\n            let signature = this.hmac (this.encode (auth), secret, 'sha512', 'base64');\n            headers['signature'] = this.decode (signature);\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (api == 'private') {\n            if ('success' in response)\n                if (!response['success'])\n                    throw new ExchangeError (this.id + ' ' + this.json (response));\n            return response;\n        }\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class btctradeua extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'btctradeua',\n            'name': 'BTC Trade UA',\n            'countries': 'UA', // Ukraine,\n            'rateLimit': 3000,\n            'has': {\n                'CORS': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27941483-79fc7350-62d9-11e7-9f61-ac47f28fcd96.jpg',\n                'api': 'https://btc-trade.com.ua/api',\n                'www': 'https://btc-trade.com.ua',\n                'doc': 'https://docs.google.com/document/d/1ocYA0yMy_RXd561sfG3qEPZ80kyll36HUxvCRe5GbhE/edit',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'deals/{symbol}',\n                        'trades/sell/{symbol}',\n                        'trades/buy/{symbol}',\n                        'japan_stat/high/{symbol}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'auth',\n                        'ask/{symbol}',\n                        'balance',\n                        'bid/{symbol}',\n                        'buy/{symbol}',\n                        'my_orders/{symbol}',\n                        'order/status/{id}',\n                        'remove/order/{id}',\n                        'sell/{symbol}',\n                    ],\n                },\n            },\n            'markets': {\n                'BCH/UAH': { 'id': 'bch_uah', 'symbol': 'BCH/UAH', 'base': 'BCH', 'quote': 'UAH' },\n                'BTC/UAH': { 'id': 'btc_uah', 'symbol': 'BTC/UAH', 'base': 'BTC', 'quote': 'UAH', 'precision': { 'price': 1 }, 'limits': { 'amount': { 'min': 0.0000000001 }}},\n                'DASH/BTC': { 'id': 'dash_btc', 'symbol': 'DASH/BTC', 'base': 'DASH', 'quote': 'BTC' },\n                'DASH/UAH': { 'id': 'dash_uah', 'symbol': 'DASH/UAH', 'base': 'DASH', 'quote': 'UAH' },\n                'DOGE/BTC': { 'id': 'doge_btc', 'symbol': 'DOGE/BTC', 'base': 'DOGE', 'quote': 'BTC' },\n                'DOGE/UAH': { 'id': 'doge_uah', 'symbol': 'DOGE/UAH', 'base': 'DOGE', 'quote': 'UAH' },\n                'ETH/UAH': { 'id': 'eth_uah', 'symbol': 'ETH/UAH', 'base': 'ETH', 'quote': 'UAH' },\n                'ITI/UAH': { 'id': 'iti_uah', 'symbol': 'ITI/UAH', 'base': 'ITI', 'quote': 'UAH' },\n                'KRB/UAH': { 'id': 'krb_uah', 'symbol': 'KRB/UAH', 'base': 'KRB', 'quote': 'UAH' },\n                'LTC/BTC': { 'id': 'ltc_btc', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC' },\n                'LTC/UAH': { 'id': 'ltc_uah', 'symbol': 'LTC/UAH', 'base': 'LTC', 'quote': 'UAH' },\n                'NVC/BTC': { 'id': 'nvc_btc', 'symbol': 'NVC/BTC', 'base': 'NVC', 'quote': 'BTC' },\n                'NVC/UAH': { 'id': 'nvc_uah', 'symbol': 'NVC/UAH', 'base': 'NVC', 'quote': 'UAH' },\n                'PPC/BTC': { 'id': 'ppc_btc', 'symbol': 'PPC/BTC', 'base': 'PPC', 'quote': 'BTC' },\n                'SIB/UAH': { 'id': 'sib_uah', 'symbol': 'SIB/UAH', 'base': 'SIB', 'quote': 'UAH' },\n                'XMR/UAH': { 'id': 'xmr_uah', 'symbol': 'XMR/UAH', 'base': 'XMR', 'quote': 'UAH' },\n                'ZEC/UAH': { 'id': 'zec_uah', 'symbol': 'ZEC/UAH', 'base': 'ZEC', 'quote': 'UAH' },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.1 / 100,\n                    'taker': 0.1 / 100,\n                },\n                'funding': {\n                    'withdraw': {\n                        'BTC': 0.0006,\n                        'LTC': 0.01,\n                        'NVC': 0.01,\n                        'DOGE': 10,\n                    },\n                },\n            },\n        });\n    }\n\n    signIn () {\n        return this.privatePostAuth ();\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostBalance ();\n        let result = { 'info': response };\n        if ('accounts' in response) {\n            let accounts = response['accounts'];\n            for (let b = 0; b < accounts.length; b++) {\n                let account = accounts[b];\n                let currency = account['currency'];\n                let balance = parseFloat (account['balance']);\n                result[currency] = {\n                    'free': balance,\n                    'used': 0.0,\n                    'total': balance,\n                };\n            }\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let market = this.market (symbol);\n        let bids = await this.publicGetTradesBuySymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        let asks = await this.publicGetTradesSellSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        let orderbook = {\n            'bids': [],\n            'asks': [],\n        };\n        if (bids) {\n            if ('list' in bids)\n                orderbook['bids'] = bids['list'];\n        }\n        if (asks) {\n            if ('list' in asks)\n                orderbook['asks'] = asks['list'];\n        }\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'currency_trade');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let response = await this.publicGetJapanStatHighSymbol (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        let orderbook = await this.fetchOrderBook (symbol);\n        let bid = undefined;\n        let numBids = orderbook['bids'].length;\n        if (numBids > 0)\n            bid = orderbook['bids'][0][0];\n        let ask = undefined;\n        let numAsks = orderbook['asks'].length;\n        if (numAsks > 0)\n            ask = orderbook['asks'][0][0];\n        let ticker = response['trades'];\n        let timestamp = this.milliseconds ();\n        let result = {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': bid,\n            'ask': ask,\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': undefined,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n        let tickerLength = ticker.length;\n        if (tickerLength > 0) {\n            let start = Math.max (tickerLength - 48, 0);\n            for (let t = start; t < ticker.length; t++) {\n                let candle = ticker[t];\n                if (typeof result['open'] == 'undefined')\n                    result['open'] = candle[1];\n                if ((typeof result['high'] == 'undefined') || (result['high'] < candle[2]))\n                    result['high'] = candle[2];\n                if ((typeof result['low'] == 'undefined') || (result['low'] > candle[3]))\n                    result['low'] = candle[3];\n                if (typeof result['baseVolume'] == 'undefined')\n                    result['baseVolume'] = -candle[5];\n                else\n                    result['baseVolume'] -= candle[5];\n            }\n            let last = tickerLength - 1;\n            result['close'] = ticker[last][4];\n            result['baseVolume'] = -1 * result['baseVolume'];\n        }\n        return result;\n    }\n\n    convertCyrillicMonthNameToString (cyrillic) {\n        let months = {\n            'января': '01',\n            'февраля': '02',\n            'марта': '03',\n            'апреля': '04',\n            'мая': '05',\n            'июня': '06',\n            'июля': '07',\n            'августа': '08',\n            'сентября': '09',\n            'октября': '10',\n            'ноября': '11',\n            'декабря': '12',\n        };\n        let month = undefined;\n        if (cyrillic in months)\n            month = months[cyrillic];\n        return month;\n    }\n\n    parseCyrillicDatetime (cyrillic) {\n        let parts = cyrillic.split (' ');\n        let day = parts[0];\n        let month = this.convertCyrillicMonthNameToString (parts[1]);\n        if (!month)\n            throw new ExchangeError (this.id + ' parseTrade() undefined month name: ' + cyrillic);\n        let year = parts[2];\n        let hms = parts[4];\n        let hmsLength = hms.length;\n        if (hmsLength == 7) {\n            hms = '0' + hms;\n        }\n        if (day.length == 1) {\n            day = '0' + day;\n        }\n        let ymd = [ year, month, day ].join ('-');\n        let ymdhms = ymd + 'T' + hms;\n        let timestamp = this.parse8601 (ymdhms);\n        // server reports local time, adjust to UTC\n        let md = [ month, day ].join ('');\n        md = parseInt (md);\n        // a special case for DST\n        // subtract 2 hours during winter\n        if (md < 325 || md > 1028)\n            return timestamp - 7200000;\n        // subtract 3 hours during summer\n        return timestamp - 10800000;\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = this.parseCyrillicDatetime (trade['pub_date']);\n        return {\n            'id': trade['id'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': 'limit',\n            'side': trade['type'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amnt_trade']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetDealsSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        // they report each trade twice (once for both of the two sides of the fill)\n        // deduplicate trades for that reason\n        let trades = [];\n        for (let i = 0; i < response.length; i++) {\n            if (response[i]['id'] % 2) {\n                trades.push (response[i]);\n            }\n        }\n        return this.parseTrades (trades, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type == 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let market = this.market (symbol);\n        let method = 'privatePost' + this.capitalize (side) + 'Id';\n        let order = {\n            'count': amount,\n            'currency1': market['quote'],\n            'currency': market['base'],\n            'price': price,\n        };\n        return this[method] (this.extend (order, params));\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostRemoveOrderId ({ 'id': id });\n    }\n\n    parseOrder (trade, market) {\n        let timestamp = this.milliseconds;\n        return {\n            'id': trade['id'],\n            'timestamp': timestamp, // until they fix their timestamp\n            'datetime': this.iso8601 (timestamp),\n            'status': 'open',\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['type'],\n            'price': trade['price'],\n            'amount': trade['amnt_trade'],\n            'filled': 0,\n            'remaining': trade['amnt_trade'],\n            'trades': undefined,\n            'info': trade,\n        };\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOpenOrders requires a symbol param');\n        let market = this.market (symbol);\n        let response = await this.privatePostMyOrdersSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        let orders = response['your_open_orders'];\n        return this.parseOrders (orders, market, since, limit);\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += this.implodeParams (path, query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({\n                'out_order_id': nonce,\n                'nonce': nonce,\n            }, query));\n            let auth = body + this.secret;\n            headers = {\n                'public-key': this.apiKey,\n                'api-sign': this.hash (this.encode (auth), 'sha256'),\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class btcturk extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'btcturk',\n            'name': 'BTCTurk',\n            'countries': 'TR', // Turkey\n            'rateLimit': 1000,\n            'has': {\n                'CORS': true,\n                'fetchTickers': true,\n                'fetchOHLCV': true,\n            },\n            'timeframes': {\n                '1d': '1d',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27992709-18e15646-64a3-11e7-9fa2-b0950ec7712f.jpg',\n                'api': 'https://www.btcturk.com/api',\n                'www': 'https://www.btcturk.com',\n                'doc': 'https://github.com/BTCTrader/broker-api-docs',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'ohlcdata', // ?last=COUNT\n                        'orderbook',\n                        'ticker',\n                        'trades',   // ?last=COUNT (max 50)\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'balance',\n                        'openOrders',\n                        'userTransactions', // ?offset=0&limit=25&sort=asc\n                    ],\n                    'post': [\n                        'buy',\n                        'cancelOrder',\n                        'sell',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/TRY': { 'id': 'BTCTRY', 'symbol': 'BTC/TRY', 'base': 'BTC', 'quote': 'TRY', 'maker': 0.002 * 1.18, 'taker': 0.0035 * 1.18 },\n                'ETH/TRY': { 'id': 'ETHTRY', 'symbol': 'ETH/TRY', 'base': 'ETH', 'quote': 'TRY', 'maker': 0.002 * 1.18, 'taker': 0.0035 * 1.18 },\n                'ETH/BTC': { 'id': 'ETHBTC', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC', 'maker': 0.002 * 1.18, 'taker': 0.0035 * 1.18 },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privateGetBalance ();\n        let result = { 'info': response };\n        let base = {\n            'free': response['bitcoin_available'],\n            'used': response['bitcoin_reserved'],\n            'total': response['bitcoin_balance'],\n        };\n        let quote = {\n            'free': response['money_available'],\n            'used': response['money_reserved'],\n            'total': response['money_balance'],\n        };\n        let symbol = this.symbols[0];\n        let market = this.markets[symbol];\n        result[market['base']] = base;\n        result[market['quote']] = quote;\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let market = this.market (symbol);\n        let orderbook = await this.publicGetOrderbook (this.extend ({\n            'pairSymbol': market['id'],\n        }, params));\n        let timestamp = parseInt (orderbook['timestamp'] * 1000);\n        return this.parseOrderBook (orderbook, timestamp);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = parseInt (ticker['timestamp']) * 1000;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': parseFloat (ticker['open']),\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': parseFloat (ticker['average']),\n            'baseVolume': parseFloat (ticker['volume']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTicker (params);\n        let result = {};\n        for (let i = 0; i < tickers.length; i++) {\n            let ticker = tickers[i];\n            let symbol = ticker['pair'];\n            let market = undefined;\n            if (symbol in this.markets_by_id) {\n                market = this.markets_by_id[symbol];\n                symbol = market['symbol'];\n            }\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.fetchTickers ();\n        let result = undefined;\n        if (symbol in tickers)\n            result = tickers[symbol];\n        return result;\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['date'] * 1000;\n        return {\n            'id': trade['tid'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        // let maxCount = 50;\n        let response = await this.publicGetTrades (this.extend ({\n            'pairSymbol': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1d', since = undefined, limit = undefined) {\n        let timestamp = this.parse8601 (ohlcv['Time']);\n        return [\n            timestamp,\n            ohlcv['Open'],\n            ohlcv['High'],\n            ohlcv['Low'],\n            ohlcv['Close'],\n            ohlcv['Volume'],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1d', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {};\n        if (limit)\n            request['last'] = limit;\n        let response = await this.publicGetOhlcdata (this.extend (request, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let method = 'privatePost' + this.capitalize (side);\n        let order = {\n            'Type': (side == 'buy') ? 'BuyBtc' : 'SelBtc',\n            'IsMarketOrder': (type == 'market') ? 1 : 0,\n        };\n        if (type == 'market') {\n            if (side == 'buy')\n                order['Total'] = amount;\n            else\n                order['Amount'] = amount;\n        } else {\n            order['Price'] = price;\n            order['Amount'] = amount;\n        }\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelOrder ({ 'id': id });\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        if (this.id == 'btctrader')\n            throw new ExchangeError (this.id + ' is an abstract base API for BTCExchange, BTCTurk');\n        let url = this.urls['api'] + '/' + path;\n        if (api == 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            body = this.urlencode (params);\n            let secret = this.base64ToBinary (this.secret);\n            let auth = this.apiKey + nonce;\n            headers = {\n                'X-PCK': this.apiKey,\n                'X-Stamp': nonce,\n                'X-Signature': this.stringToBase64(this.hmac (this.encode (auth), secret, 'sha256', 'binary')),\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class btcx extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'btcx',\n            'name': 'BTCX',\n            'countries': [ 'IS', 'US', 'EU' ],\n            'rateLimit': 1500, // support in english is very poor, unable to tell rate limits\n            'version': 'v1',\n            'has': {\n                'CORS': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766385-9fdcc98c-5ed6-11e7-8f14-66d5e5cd47e6.jpg',\n                'api': 'https://btc-x.is/api',\n                'www': 'https://btc-x.is',\n                'doc': 'https://btc-x.is/custom/api-document.html',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'depth/{id}/{limit}',\n                        'ticker/{id}',\n                        'trade/{id}/{limit}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balance',\n                        'cancel',\n                        'history',\n                        'order',\n                        'redeem',\n                        'trade',\n                        'withdraw',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/USD': { 'id': 'btc/usd', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD' },\n                'BTC/EUR': { 'id': 'btc/eur', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR' },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let balances = await this.privatePostBalance ();\n        let result = { 'info': balances };\n        let currencies = Object.keys (balances);\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let uppercase = currency.toUpperCase ();\n            let account = {\n                'free': balances[currency],\n                'used': 0.0,\n                'total': balances[currency],\n            };\n            result[uppercase] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetDepthIdLimit (this.extend ({\n            'id': this.marketId (symbol),\n            'limit': 1000,\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'amount');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetTickerId (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        let timestamp = ticker['time'] * 1000;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['sell']),\n            'ask': parseFloat (ticker['buy']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': parseFloat (ticker['volume']),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        let side = (trade['type'] == 'ask') ? 'sell' : 'buy';\n        return {\n            'id': trade['id'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': side,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetTradeIdLimit (this.extend ({\n            'id': market['id'],\n            'limit': 1000,\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let response = await this.privatePostTrade (this.extend ({\n            'type': side.toUpperCase (),\n            'market': this.marketId (symbol),\n            'amount': amount,\n            'price': price,\n        }, params));\n        return {\n            'info': response,\n            'id': response['order']['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancel ({ 'order': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/';\n        if (api == 'public') {\n            url += this.implodeParams (path, params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            url += api;\n            body = this.urlencode (this.extend ({\n                'Method': path.toUpperCase (),\n                'Nonce': nonce,\n            }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Key': this.apiKey,\n                'Signature': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bter extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bter',\n            'name': 'Bter',\n            'countries': [ 'VG', 'CN' ], // British Virgin Islands, China\n            'version': '2',\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27980479-cfa3188c-6387-11e7-8191-93fc4184ba5c.jpg',\n                'api': {\n                    'public': 'https://data.bter.com/api',\n                    'private': 'https://api.bter.com/api',\n                },\n                'www': 'https://bter.com',\n                'doc': 'https://bter.com/api2',\n                'fees': 'https://bter.com/fee',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'pairs',\n                        'marketinfo',\n                        'marketlist',\n                        'tickers',\n                        'ticker/{id}',\n                        'orderBook/{id}',\n                        'trade/{id}',\n                        'tradeHistory/{id}',\n                        'tradeHistory/{id}/{tid}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balances',\n                        'depositAddress',\n                        'newAddress',\n                        'depositsWithdrawals',\n                        'buy',\n                        'sell',\n                        'cancelOrder',\n                        'cancelAllOrders',\n                        'getOrder',\n                        'openOrders',\n                        'tradeHistory',\n                        'withdraw',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicGetMarketinfo ();\n        let markets = this.safeValue (response, 'pairs');\n        if (!markets)\n            throw new ExchangeError (this.id + ' fetchMarkets got an unrecognized response');\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let keys = Object.keys (market);\n            let id = keys[0];\n            let details = market[id];\n            let [ base, quote ] = id.split ('_');\n            base = base.toUpperCase ();\n            quote = quote.toUpperCase ();\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            let precision = {\n                'amount': details['decimal_places'],\n                'price': details['decimal_places'],\n            };\n            let amountLimits = {\n                'min': details['min_amount'],\n                'max': undefined,\n            };\n            let priceLimits = {\n                'min': undefined,\n                'max': undefined,\n            };\n            let limits = {\n                'amount': amountLimits,\n                'price': priceLimits,\n            };\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n                'maker': details['fee'] / 100,\n                'taker': details['fee'] / 100,\n                'precision': precision,\n                'limits': limits,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balance = await this.privatePostBalances ();\n        let result = { 'info': balance };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let code = this.commonCurrencyCode (currency);\n            let account = this.account ();\n            if ('available' in balance) {\n                if (currency in balance['available']) {\n                    account['free'] = parseFloat (balance['available'][currency]);\n                }\n            }\n            if ('locked' in balance) {\n                if (currency in balance['locked']) {\n                    account['used'] = parseFloat (balance['locked'][currency]);\n                }\n            }\n            account['total'] = this.sum (account['free'], account['used']);\n            result[code] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderBookId (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        let result = this.parseOrderBook (orderbook);\n        result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high24hr']),\n            'low': parseFloat (ticker['low24hr']),\n            'bid': parseFloat (ticker['highestBid']),\n            'ask': parseFloat (ticker['lowestAsk']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': parseFloat (ticker['percentChange']),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['quoteVolume']),\n            'quoteVolume': parseFloat (ticker['baseVolume']),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTickers (params);\n        let result = {};\n        let ids = Object.keys (tickers);\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let [ baseId, quoteId ] = id.split ('_');\n            let base = baseId.toUpperCase ();\n            let quote = quoteId.toUpperCase ();\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            let ticker = tickers[id];\n            let market = undefined;\n            if (symbol in this.markets)\n                market = this.markets[symbol];\n            if (id in this.markets_by_id)\n                market = this.markets_by_id[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetTickerId (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = this.parse8601 (trade['date']);\n        return {\n            'id': trade['tradeID'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['type'],\n            'price': trade['rate'],\n            'amount': this.safeFloat (trade, 'amount'),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTradeHistoryId (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTrades (response['data'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type === 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        await this.loadMarkets ();\n        let method = 'privatePost' + this.capitalize (side);\n        let order = {\n            'currencyPair': this.marketId (symbol),\n            'rate': price,\n            'amount': amount,\n        };\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['orderNumber'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelOrder ({ 'orderNumber': id });\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostWithdraw (this.extend ({\n            'currency': currency.toLowerCase (),\n            'amount': amount,\n            'address': address, // Address must exist in you AddressBook in security settings\n        }, params));\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let prefix = (api === 'private') ? (api + '/') : '';\n        let url = this.urls['api'][api] + this.version + '/1/' + prefix + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let request = { 'nonce': nonce };\n            body = this.urlencode (this.extend (request, query));\n            let signature = this.hmac (this.encode (body), this.encode (this.secret), 'sha512');\n            headers = {\n                'Key': this.apiKey,\n                'Sign': signature,\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('result' in response)\n            if (response['result'] !== 'true')\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class bxinth extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'bxinth',\n            'name': 'BX.in.th',\n            'countries': 'TH', // Thailand\n            'rateLimit': 1500,\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766412-567b1eb4-5ed7-11e7-94a8-ff6a3884f6c5.jpg',\n                'api': 'https://bx.in.th/api',\n                'www': 'https://bx.in.th',\n                'doc': 'https://bx.in.th/info/api',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        '', // ticker\n                        'options',\n                        'optionbook',\n                        'orderbook',\n                        'pairing',\n                        'trade',\n                        'tradehistory',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balance',\n                        'biller',\n                        'billgroup',\n                        'billpay',\n                        'cancel',\n                        'deposit',\n                        'getorders',\n                        'history',\n                        'option-issue',\n                        'option-bid',\n                        'option-sell',\n                        'option-myissue',\n                        'option-mybid',\n                        'option-myoptions',\n                        'option-exercise',\n                        'option-cancel',\n                        'option-history',\n                        'order',\n                        'withdrawal',\n                        'withdrawal-history',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'taker': 0.25 / 100,\n                    'maker': 0.25 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetPairing ();\n        let keys = Object.keys (markets);\n        let result = [];\n        for (let p = 0; p < keys.length; p++) {\n            let market = markets[keys[p]];\n            let id = market['pairing_id'].toString ();\n            let base = market['secondary_currency'];\n            let quote = market['primary_currency'];\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    commonCurrencyCode (currency) {\n        // why would they use three letters instead of four for currency codes\n        if (currency == 'DAS')\n            return 'DASH';\n        if (currency == 'DOG')\n            return 'DOGE';\n        return currency;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostBalance ();\n        let balance = response['balance'];\n        let result = { 'info': balance };\n        let currencies = Object.keys (balance);\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let code = this.commonCurrencyCode (currency);\n            let account = {\n                'free': parseFloat (balance[currency]['available']),\n                'used': 0.0,\n                'total': parseFloat (balance[currency]['total']),\n            };\n            account['used'] = account['total'] - account['free'];\n            result[code] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderbook (this.extend ({\n            'pairing': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['orderbook']['bids']['highbid']),\n            'ask': parseFloat (ticker['orderbook']['asks']['highbid']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last_price']),\n            'change': parseFloat (ticker['change']),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume_24hours']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGet (params);\n        let result = {};\n        let ids = Object.keys (tickers);\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let ticker = tickers[id];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let tickers = await this.publicGet (this.extend ({\n            'pairing': market['id'],\n        }, params));\n        let id = market['id'].toString ();\n        let ticker = tickers[id];\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = this.parse8601 (trade['trade_date']);\n        return {\n            'id': trade['trade_id'],\n            'info': trade,\n            'order': trade['order_id'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['trade_type'],\n            'price': parseFloat (trade['rate']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTrade (this.extend ({\n            'pairing': market['id'],\n        }, params));\n        return this.parseTrades (response['trades'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostOrder (this.extend ({\n            'pairing': this.marketId (symbol),\n            'type': side,\n            'amount': amount,\n            'rate': price,\n        }, params));\n        return {\n            'info': response,\n            'id': response['order_id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let pairing = undefined; // TODO fixme\n        return await this.privatePostCancel ({\n            'order_id': id,\n            'pairing': pairing,\n        });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/';\n        if (path)\n            url += path + '/';\n        if (Object.keys (params).length)\n            url += '?' + this.urlencode (params);\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let auth = this.apiKey + nonce.toString () + this.secret;\n            let signature = this.hash (this.encode (auth), 'sha256');\n            body = this.urlencode (this.extend ({\n                'key': this.apiKey,\n                'nonce': nonce,\n                'signature': signature,\n                // twofa: this.twofa,\n            }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (api == 'public')\n            return response;\n        if ('success' in response)\n            if (response['success'])\n                return response;\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class ccex extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'ccex',\n            'name': 'C-CEX',\n            'countries': [ 'DE', 'EU' ],\n            'rateLimit': 1500,\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766433-16881f90-5ed8-11e7-92f8-3d92cc747a6c.jpg',\n                'api': {\n                    'tickers': 'https://c-cex.com/t',\n                    'public': 'https://c-cex.com/t/api_pub.html',\n                    'private': 'https://c-cex.com/t/api.html',\n                },\n                'www': 'https://c-cex.com',\n                'doc': 'https://c-cex.com/?id=api',\n            },\n            'api': {\n                'tickers': {\n                    'get': [\n                        'coinnames',\n                        '{market}',\n                        'pairs',\n                        'prices',\n                        'volume_{coin}',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'balancedistribution',\n                        'markethistory',\n                        'markets',\n                        'marketsummaries',\n                        'orderbook',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'buylimit',\n                        'cancel',\n                        'getbalance',\n                        'getbalances',\n                        'getopenorders',\n                        'getorder',\n                        'getorderhistory',\n                        'mytrades',\n                        'selllimit',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'taker': 0.2 / 100,\n                    'maker': 0.2 / 100,\n                },\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        if (currency == 'IOT')\n            return 'IoTcoin';\n        if (currency == 'BLC')\n            return 'Cryptobullcoin';\n        if (currency == 'XID')\n            return 'InternationalDiamond';\n        return currency;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetMarkets ();\n        let result = [];\n        for (let p = 0; p < markets['result'].length; p++) {\n            let market = markets['result'][p];\n            let id = market['MarketName'];\n            let base = market['MarketCurrency'];\n            let quote = market['BaseCurrency'];\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            }));\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetBalances ();\n        let balances = response['result'];\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let code = balance['Currency'];\n            let currency = this.commonCurrencyCode (code);\n            let account = {\n                'free': balance['Available'],\n                'used': balance['Pending'],\n                'total': balance['Balance'],\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetOrderbook (this.extend ({\n            'market': this.marketId (symbol),\n            'type': 'both',\n            'depth': 100,\n        }, params));\n        let orderbook = response['result'];\n        return this.parseOrderBook (orderbook, undefined, 'buy', 'sell', 'Rate', 'Quantity');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['updated'] * 1000;\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy']),\n            'ask': parseFloat (ticker['sell']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['lastprice']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': parseFloat (ticker['avg']),\n            'baseVolume': undefined,\n            'quoteVolume': this.safeFloat (ticker, 'buysupport'),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.tickersGetPrices (params);\n        let result = { 'info': tickers };\n        let ids = Object.keys (tickers);\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let ticker = tickers[id];\n            let uppercase = id.toUpperCase ();\n            let market = undefined;\n            let symbol = undefined;\n            if (uppercase in this.markets_by_id) {\n                market = this.markets_by_id[uppercase];\n                symbol = market['symbol'];\n            } else {\n                let [ base, quote ] = uppercase.split ('-');\n                base = this.commonCurrencyCode (base);\n                quote = this.commonCurrencyCode (quote);\n                symbol = base + '/' + quote;\n            }\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.tickersGetMarket (this.extend ({\n            'market': market['id'].toLowerCase (),\n        }, params));\n        let ticker = response['ticker'];\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = this.parse8601 (trade['TimeStamp']);\n        return {\n            'id': trade['Id'],\n            'info': trade,\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['OrderType'].toLowerCase (),\n            'price': trade['Price'],\n            'amount': trade['Quantity'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetMarkethistory (this.extend ({\n            'market': market['id'],\n            'type': 'both',\n            'depth': 100,\n        }, params));\n        return this.parseTrades (response['result'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let method = 'privateGet' + this.capitalize (side) + type;\n        let response = await this[method] (this.extend ({\n            'market': this.marketId (symbol),\n            'quantity': amount,\n            'rate': price,\n        }, params));\n        return {\n            'info': response,\n            'id': response['result']['uuid'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privateGetCancel ({ 'uuid': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let query = this.keysort (this.extend ({\n                'a': path,\n                'apikey': this.apiKey,\n                'nonce': nonce,\n            }, params));\n            url += '?' + this.urlencode (query);\n            headers = { 'apisign': this.hmac (this.encode (url), this.encode (this.secret), 'sha512') };\n        } else if (api == 'public') {\n            url += '?' + this.urlencode (this.extend ({\n                'a': 'get' + path,\n            }, params));\n        } else {\n            url += '/' + this.implodeParams (path, params) + '.json';\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (api == 'tickers')\n            return response;\n        if ('success' in response)\n            if (response['success'])\n                return response;\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InvalidOrder } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class cex extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'cex',\n            'name': 'CEX.IO',\n            'countries': [ 'GB', 'EU', 'CY', 'RU' ],\n            'rateLimit': 1500,\n            'has': {\n                'CORS': true,\n                'fetchTickers': true,\n                'fetchOHLCV': true,\n                'fetchOpenOrders': true,\n            },\n            'timeframes': {\n                '1m': '1m',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766442-8ddc33b0-5ed8-11e7-8b98-f786aef0f3c9.jpg',\n                'api': 'https://cex.io/api',\n                'www': 'https://cex.io',\n                'doc': 'https://cex.io/cex-api',\n                'fees': [\n                    'https://cex.io/fee-schedule',\n                    'https://cex.io/limits-commissions',\n                ],\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'uid': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'currency_limits/',\n                        'last_price/{pair}/',\n                        'last_prices/{currencies}/',\n                        'ohlcv/hd/{yyyymmdd}/{pair}',\n                        'order_book/{pair}/',\n                        'ticker/{pair}/',\n                        'tickers/{currencies}/',\n                        'trade_history/{pair}/',\n                    ],\n                    'post': [\n                        'convert/{pair}',\n                        'price_stats/{pair}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'active_orders_status/',\n                        'archived_orders/{pair}/',\n                        'balance/',\n                        'cancel_order/',\n                        'cancel_orders/{pair}/',\n                        'cancel_replace_order/{pair}/',\n                        'close_position/{pair}/',\n                        'get_address/',\n                        'get_myfee/',\n                        'get_order/',\n                        'get_order_tx/',\n                        'open_orders/{pair}/',\n                        'open_orders/',\n                        'open_position/{pair}/',\n                        'open_positions/{pair}/',\n                        'place_order/{pair}/',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.16 / 100,\n                    'taker': 0.25 / 100,\n                },\n                'funding': {\n                    'withdraw': {\n                        // 'USD': undefined,\n                        // 'EUR': undefined,\n                        // 'RUB': undefined,\n                        // 'GBP': undefined,\n                        'BTC': 0.001,\n                        'ETH': 0.01,\n                        'BCH': 0.001,\n                        'DASH': 0.01,\n                        'BTG': 0.001,\n                        'ZEC': 0.001,\n                        'XRP': 0.02,\n                        'XLM': undefined,\n                    },\n                    'deposit': {\n                        // 'USD': amount => amount * 0.035 + 0.25,\n                        // 'EUR': amount => amount * 0.035 + 0.24,\n                        // 'RUB': amount => amount * 0.05 + 15.57,\n                        // 'GBP': amount => amount * 0.035 + 0.2,\n                        'BTC': 0.0,\n                        'ETH': 0.0,\n                        'BCH': 0.0,\n                        'DASH': 0.0,\n                        'BTG': 0.0,\n                        'ZEC': 0.0,\n                        'XRP': 0.0,\n                        'XLM': 0.0,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetCurrencyLimits ();\n        let result = [];\n        for (let p = 0; p < markets['data']['pairs'].length; p++) {\n            let market = markets['data']['pairs'][p];\n            let id = market['symbol1'] + '/' + market['symbol2'];\n            let symbol = id;\n            let [ base, quote ] = symbol.split ('/');\n            result.push ({\n                'id': id,\n                'info': market,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'precision': {\n                    'price': this.precisionFromString (market['minPrice']),\n                    'amount': -1 * Math.log10 (market['minLotSize']),\n                },\n                'limits': {\n                    'amount': {\n                        'min': market['minLotSize'],\n                        'max': market['maxLotSize'],\n                    },\n                    'price': {\n                        'min': parseFloat (market['minPrice']),\n                        'max': parseFloat (market['maxPrice']),\n                    },\n                    'cost': {\n                        'min': market['minLotSizeS2'],\n                        'max': undefined,\n                    },\n                },\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostBalance ();\n        let result = { 'info': response };\n        let ommited = [ 'username', 'timestamp' ];\n        let balances = this.omit (response, ommited);\n        let currencies = Object.keys (balances);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            if (currency in balances) {\n                let account = {\n                    'free': this.safeFloat (balances[currency], 'available', 0.0),\n                    'used': this.safeFloat (balances[currency], 'orders', 0.0),\n                    'total': 0.0,\n                };\n                account['total'] = this.sum (account['free'], account['used']);\n                result[currency] = account;\n            }\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderBookPair (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        let timestamp = orderbook['timestamp'] * 1000;\n        return this.parseOrderBook (orderbook, timestamp);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv[0] * 1000,\n            ohlcv[1],\n            ohlcv[2],\n            ohlcv[3],\n            ohlcv[4],\n            ohlcv[5],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        if (!since)\n            since = this.milliseconds () - 86400000; // yesterday\n        let ymd = this.Ymd (since);\n        ymd = ymd.split ('-');\n        ymd = ymd.join ('');\n        let request = {\n            'pair': market['id'],\n            'yyyymmdd': ymd,\n        };\n        let response = await this.publicGetOhlcvHdYyyymmddPair (this.extend (request, params));\n        let key = 'data' + this.timeframes[timeframe];\n        let ohlcvs = JSON.parse (response[key]);\n        return this.parseOHLCVs (ohlcvs, market, timeframe, since, limit);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = undefined;\n        let iso8601 = undefined;\n        if ('timestamp' in ticker) {\n            timestamp = parseInt (ticker['timestamp']) * 1000;\n            iso8601 = this.iso8601 (timestamp);\n        }\n        let volume = this.safeFloat (ticker, 'volume');\n        let high = this.safeFloat (ticker, 'high');\n        let low = this.safeFloat (ticker, 'low');\n        let bid = this.safeFloat (ticker, 'bid');\n        let ask = this.safeFloat (ticker, 'ask');\n        let last = this.safeFloat (ticker, 'last');\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': iso8601,\n            'high': high,\n            'low': low,\n            'bid': bid,\n            'ask': ask,\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': last,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': volume,\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let currencies = Object.keys (this.currencies);\n        let response = await this.publicGetTickersCurrencies (this.extend ({\n            'currencies': currencies.join ('/'),\n        }, params));\n        let tickers = response['data'];\n        let result = {};\n        for (let t = 0; t < tickers.length; t++) {\n            let ticker = tickers[t];\n            let symbol = ticker['pair'].replace (':', '/');\n            let market = this.markets[symbol];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetTickerPair (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        return {\n            'info': trade,\n            'id': trade['tid'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['type'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTradeHistoryPair (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'pair': this.marketId (symbol),\n            'type': side,\n            'amount': amount,\n        };\n        if (type === 'limit') {\n            order['price'] = price;\n        } else {\n            // for market buy CEX.io requires the amount of quote currency to spend\n            if (side === 'buy') {\n                if (!price) {\n                    throw new InvalidOrder ('For market buy orders ' + this.id + \" requires the amount of quote currency to spend, to calculate proper costs call createOrder (symbol, 'market', 'buy', amount, price)\");\n                }\n                order['amount'] = amount * price;\n            }\n            order['order_type'] = type;\n        }\n        let response = await this.privatePostPlaceOrderPair (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelOrder ({ 'id': id });\n    }\n\n    parseOrder (order, market = undefined) {\n        let timestamp = parseInt (order['time']);\n        let symbol = undefined;\n        if (!market) {\n            let symbol = order['symbol1'] + '/' + order['symbol2'];\n            if (symbol in this.markets)\n                market = this.market (symbol);\n        }\n        let status = order['status'];\n        if (status === 'a') {\n            status = 'open'; // the unified status\n        } else if (status === 'cd') {\n            status = 'canceled';\n        } else if (status === 'c') {\n            status = 'canceled';\n        } else if (status === 'd') {\n            status = 'closed';\n        }\n        let price = this.safeFloat (order, 'price');\n        let amount = this.safeFloat (order, 'amount');\n        let remaining = this.safeFloat (order, 'pending');\n        if (!remaining)\n            remaining = this.safeFloat (order, 'remains');\n        let filled = amount - remaining;\n        let fee = undefined;\n        let cost = undefined;\n        if (market) {\n            symbol = market['symbol'];\n            cost = this.safeFloat (order, 'ta:' + market['quote']);\n            let baseFee = 'fa:' + market['base'];\n            let quoteFee = 'fa:' + market['quote'];\n            let feeRate = this.safeFloat (order, 'tradingFeeMaker');\n            if (!feeRate)\n                feeRate = this.safeFloat (order, 'tradingFeeTaker', feeRate);\n            if (feeRate)\n                feeRate /= 100.0; // convert to mathematically-correct percentage coefficients: 1.0 = 100%\n            if (baseFee in order) {\n                fee = {\n                    'currency': market['base'],\n                    'rate': feeRate,\n                    'cost': this.safeFloat (order, baseFee),\n                };\n            } else if (quoteFee in order) {\n                fee = {\n                    'currency': market['quote'],\n                    'rate': feeRate,\n                    'cost': this.safeFloat (order, quoteFee),\n                };\n            }\n        }\n        if (!cost)\n            cost = price * filled;\n        return {\n            'id': order['id'],\n            'datetime': this.iso8601 (timestamp),\n            'timestamp': timestamp,\n            'status': status,\n            'symbol': symbol,\n            'type': undefined,\n            'side': order['type'],\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'trades': undefined,\n            'fee': fee,\n            'info': order,\n        };\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        let method = 'privatePostOpenOrders';\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['pair'] = market['id'];\n            method += 'Pair';\n        }\n        let orders = await this[method] (this.extend (request, params));\n        for (let i = 0; i < orders.length; i++) {\n            orders[i] = this.extend (orders[i], { 'status': 'open' });\n        }\n        return this.parseOrders (orders, market, since, limit);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetOrder (this.extend ({\n            'id': id.toString (),\n        }, params));\n        return this.parseOrder (response);\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let auth = nonce + this.uid + this.apiKey;\n            let signature = this.hmac (this.encode (auth), this.encode (this.secret));\n            body = this.urlencode (this.extend ({\n                'key': this.apiKey,\n                'signature': signature.toUpperCase (),\n                'nonce': nonce,\n            }, query));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (!response) {\n            throw new ExchangeError (this.id + ' returned ' + this.json (response));\n        } else if (response === true) {\n            return response;\n        } else if ('e' in response) {\n            if ('ok' in response)\n                if (response['ok'] === 'ok')\n                    return response;\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        } else if ('error' in response) {\n            if (response['error'])\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        return response;\n    }\n};\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst zb = require ('./zb.js');\nconst { ExchangeError } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class chbtc extends zb {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'chbtc',\n            'name': 'CHBTC',\n            'countries': 'CN',\n            'rateLimit': 1000,\n            'version': 'v1',\n            'has': {\n                'CORS': false,\n                'fetchOrder': true\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28555659-f0040dc2-7109-11e7-9d99-688a438bf9f4.jpg',\n                'api': {\n                    'public': 'http://api.chbtc.com/data', // no https for public API\n                    'private': 'https://trade.chbtc.com/api',\n                },\n                'www': 'https://trade.chbtc.com/api',\n                'doc': 'https://www.chbtc.com/i/developer',\n            },\n        });\n    }\n\n    getMarketFieldName () {\n        return 'currency';\n    }\n\n    async fetchMarkets () {\n        return {\n            'BTC/CNY': { 'id': 'btc_cny', 'symbol': 'BTC/CNY', 'base': 'BTC', 'quote': 'CNY' },\n            'LTC/CNY': { 'id': 'ltc_cny', 'symbol': 'LTC/CNY', 'base': 'LTC', 'quote': 'CNY' },\n            'ETH/CNY': { 'id': 'eth_cny', 'symbol': 'ETH/CNY', 'base': 'ETH', 'quote': 'CNY' },\n            'ETC/CNY': { 'id': 'etc_cny', 'symbol': 'ETC/CNY', 'base': 'ETC', 'quote': 'CNY' },\n            'BTS/CNY': { 'id': 'bts_cny', 'symbol': 'BTS/CNY', 'base': 'BTS', 'quote': 'CNY' },\n            // 'EOS/CNY': { 'id': 'eos_cny', 'symbol': 'EOS/CNY', 'base': 'EOS', 'quote': 'CNY' },\n            'BCH/CNY': { 'id': 'bcc_cny', 'symbol': 'BCH/CNY', 'base': 'BCH', 'quote': 'CNY' },\n            'HSR/CNY': { 'id': 'hsr_cny', 'symbol': 'HSR/CNY', 'base': 'HSR', 'quote': 'CNY' },\n            'QTUM/CNY': { 'id': 'qtum_cny', 'symbol': 'QTUM/CNY', 'base': 'QTUM', 'quote': 'CNY' },\n        };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (api == 'private') {\n            if ('code' in response)\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        if ('result' in response) {\n            if (!response['result'])\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        return response;\n    }\n\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst foxbit = require ('./foxbit.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class chilebit extends foxbit {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'chilebit',\n            'name': 'ChileBit',\n            'countries': 'CL',\n            'has': {\n                'CORS': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27991414-1298f0d8-647f-11e7-9c40-d56409266336.jpg',\n                'api': {\n                    'public': 'https://api.blinktrade.com/api',\n                    'private': 'https://api.blinktrade.com/tapi',\n                },\n                'www': 'https://chilebit.net',\n                'doc': 'https://blinktrade.com/docs',\n            },\n        });\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, NotSupported } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class coincheck extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'coincheck',\n            'name': 'coincheck',\n            'countries': [ 'JP', 'ID' ],\n            'rateLimit': 1500,\n            'has': {\n                'CORS': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766464-3b5c3c74-5ed9-11e7-840e-31b32968e1da.jpg',\n                'api': 'https://coincheck.com/api',\n                'www': 'https://coincheck.com',\n                'doc': 'https://coincheck.com/documents/exchange/api',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'exchange/orders/rate',\n                        'order_books',\n                        'rate/{pair}',\n                        'ticker',\n                        'trades',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'accounts',\n                        'accounts/balance',\n                        'accounts/leverage_balance',\n                        'bank_accounts',\n                        'deposit_money',\n                        'exchange/orders/opens',\n                        'exchange/orders/transactions',\n                        'exchange/orders/transactions_pagination',\n                        'exchange/leverage/positions',\n                        'lending/borrows/matches',\n                        'send_money',\n                        'withdraws',\n                    ],\n                    'post': [\n                        'bank_accounts',\n                        'deposit_money/{id}/fast',\n                        'exchange/orders',\n                        'exchange/transfers/to_leverage',\n                        'exchange/transfers/from_leverage',\n                        'lending/borrows',\n                        'lending/borrows/{id}/repay',\n                        'send_money',\n                        'withdraws',\n                    ],\n                    'delete': [\n                        'bank_accounts/{id}',\n                        'exchange/orders/{id}',\n                        'withdraws/{id}',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/JPY': { 'id': 'btc_jpy', 'symbol': 'BTC/JPY', 'base': 'BTC', 'quote': 'JPY' }, // the only real pair\n                // 'ETH/JPY': { 'id': 'eth_jpy', 'symbol': 'ETH/JPY', 'base': 'ETH', 'quote': 'JPY' },\n                // 'ETC/JPY': { 'id': 'etc_jpy', 'symbol': 'ETC/JPY', 'base': 'ETC', 'quote': 'JPY' },\n                // 'DAO/JPY': { 'id': 'dao_jpy', 'symbol': 'DAO/JPY', 'base': 'DAO', 'quote': 'JPY' },\n                // 'LSK/JPY': { 'id': 'lsk_jpy', 'symbol': 'LSK/JPY', 'base': 'LSK', 'quote': 'JPY' },\n                // 'FCT/JPY': { 'id': 'fct_jpy', 'symbol': 'FCT/JPY', 'base': 'FCT', 'quote': 'JPY' },\n                // 'XMR/JPY': { 'id': 'xmr_jpy', 'symbol': 'XMR/JPY', 'base': 'XMR', 'quote': 'JPY' },\n                // 'REP/JPY': { 'id': 'rep_jpy', 'symbol': 'REP/JPY', 'base': 'REP', 'quote': 'JPY' },\n                // 'XRP/JPY': { 'id': 'xrp_jpy', 'symbol': 'XRP/JPY', 'base': 'XRP', 'quote': 'JPY' },\n                // 'ZEC/JPY': { 'id': 'zec_jpy', 'symbol': 'ZEC/JPY', 'base': 'ZEC', 'quote': 'JPY' },\n                // 'XEM/JPY': { 'id': 'xem_jpy', 'symbol': 'XEM/JPY', 'base': 'XEM', 'quote': 'JPY' },\n                // 'LTC/JPY': { 'id': 'ltc_jpy', 'symbol': 'LTC/JPY', 'base': 'LTC', 'quote': 'JPY' },\n                // 'DASH/JPY': { 'id': 'dash_jpy', 'symbol': 'DASH/JPY', 'base': 'DASH', 'quote': 'JPY' },\n                // 'ETH/BTC': { 'id': 'eth_btc', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC' },\n                // 'ETC/BTC': { 'id': 'etc_btc', 'symbol': 'ETC/BTC', 'base': 'ETC', 'quote': 'BTC' },\n                // 'LSK/BTC': { 'id': 'lsk_btc', 'symbol': 'LSK/BTC', 'base': 'LSK', 'quote': 'BTC' },\n                // 'FCT/BTC': { 'id': 'fct_btc', 'symbol': 'FCT/BTC', 'base': 'FCT', 'quote': 'BTC' },\n                // 'XMR/BTC': { 'id': 'xmr_btc', 'symbol': 'XMR/BTC', 'base': 'XMR', 'quote': 'BTC' },\n                // 'REP/BTC': { 'id': 'rep_btc', 'symbol': 'REP/BTC', 'base': 'REP', 'quote': 'BTC' },\n                // 'XRP/BTC': { 'id': 'xrp_btc', 'symbol': 'XRP/BTC', 'base': 'XRP', 'quote': 'BTC' },\n                // 'ZEC/BTC': { 'id': 'zec_btc', 'symbol': 'ZEC/BTC', 'base': 'ZEC', 'quote': 'BTC' },\n                // 'XEM/BTC': { 'id': 'xem_btc', 'symbol': 'XEM/BTC', 'base': 'XEM', 'quote': 'BTC' },\n                // 'LTC/BTC': { 'id': 'ltc_btc', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC' },\n                // 'DASH/BTC': { 'id': 'dash_btc', 'symbol': 'DASH/BTC', 'base': 'DASH', 'quote': 'BTC' },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let balances = await this.privateGetAccountsBalance ();\n        let result = { 'info': balances };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let account = this.account ();\n            if (lowercase in balances)\n                account['free'] = parseFloat (balances[lowercase]);\n            let reserved = lowercase + '_reserved';\n            if (reserved in balances)\n                account['used'] = parseFloat (balances[reserved]);\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        if (symbol != 'BTC/JPY')\n            throw new NotSupported (this.id + ' fetchOrderBook () supports BTC/JPY only');\n        let orderbook = await this.publicGetOrderBooks (params);\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        if (symbol != 'BTC/JPY')\n            throw new NotSupported (this.id + ' fetchTicker () supports BTC/JPY only');\n        let ticker = await this.publicGetTicker (params);\n        let timestamp = ticker['timestamp'] * 1000;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = this.parse8601 (trade['created_at']);\n        return {\n            'id': trade['id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['order_type'],\n            'price': parseFloat (trade['rate']),\n            'amount': parseFloat (trade['amount']),\n            'info': trade,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        if (symbol != 'BTC/JPY')\n            throw new NotSupported (this.id + ' fetchTrades () supports BTC/JPY only');\n        let market = this.market (symbol);\n        let response = await this.publicGetTrades (this.extend ({\n            'pair': market['id'],\n        }, params));\n        if ('success' in response)\n            if (response['success'])\n                if (typeof response['data'] !== 'undefined')\n                    return this.parseTrades (response['data'], market, since, limit);\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let order = {\n            'pair': this.marketId (symbol),\n        };\n        if (type == 'market') {\n            let order_type = type + '_' + side;\n            order['order_type'] = order_type;\n            let prefix = (side == 'buy') ? (order_type + '_') : '';\n            order[prefix + 'amount'] = amount;\n        } else {\n            order['order_type'] = side;\n            order['rate'] = price;\n            order['amount'] = amount;\n        }\n        let response = await this.privatePostExchangeOrders (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privateDeleteExchangeOrdersId ({ 'id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let queryString = '';\n            if (method == 'GET') {\n                if (Object.keys (query).length)\n                    url += '?' + this.urlencode (this.keysort (query));\n            } else {\n                if (Object.keys (query).length) {\n                    body = this.urlencode (this.keysort (query));\n                    queryString = body;\n                }\n            }\n            let auth = nonce + url + queryString;\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'ACCESS-KEY': this.apiKey,\n                'ACCESS-NONCE': nonce,\n                'ACCESS-SIGNATURE': this.hmac (this.encode (auth), this.encode (this.secret)),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (api == 'public')\n            return response;\n        if ('success' in response)\n            if (response['success'])\n                return response;\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class coinexchange extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'coinexchange',\n            'name': 'CoinExchange',\n            'countries': [ 'IN', 'JP', 'KR', 'VN', 'US' ],\n            'rateLimit': 1000,\n            // new metainfo interface\n            'has': {\n                'privateAPI': false,\n                'fetchTrades': false,\n                'fetchCurrencies': true,\n                'fetchTickers': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/34842303-29c99fca-f71c-11e7-83c1-09d900cb2334.jpg',\n                'api': 'https://www.coinexchange.io/api/v1',\n                'www': 'https://www.coinexchange.io',\n                'doc': 'https://coinexchangeio.github.io/slate/',\n                'fees': 'https://www.coinexchange.io/fees',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'getcurrency',\n                        'getcurrencies',\n                        'getmarkets',\n                        'getmarketsummaries',\n                        'getmarketsummary',\n                        'getorderbook',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.0015,\n                    'taker': 0.0015,\n                },\n            },\n            'precision': {\n                'amount': 8,\n                'price': 8,\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        return currency;\n    }\n\n    async fetchCurrencies (params = {}) {\n        let currencies = await this.publicGetCurrencies (params);\n        let precision = this.precision['amount'];\n        let result = {};\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let id = currency['CurrencyID'];\n            let code = this.commonCurrencyCode (currency['TickerCode']);\n            let active = currency['WalletStatus'] == 'online';\n            let status = 'ok';\n            if (!active)\n                status = 'disabled';\n            result[code] = {\n                'id': id,\n                'code': code,\n                'name': currency['Name'],\n                'active': active,\n                'status': status,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': undefined,\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': undefined,\n                        'max': Math.pow (10, precision),\n                    },\n                },\n                'info': currency,\n            };\n        }\n        return result;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetMarkets ();\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let id = market['MarketID'];\n            let base = this.commonCurrencyCode (market['MarketAssetCode']);\n            let quote = this.commonCurrencyCode (market['BaseCurrencyCode']);\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': true,\n                'lot': undefined,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        if (!market) {\n            let marketId = ticker['MarketID'];\n            market = this.marketsById[marketId];\n        }\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['HighPrice']),\n            'low': parseFloat (ticker['LowPrice']),\n            'bid': parseFloat (ticker['BidPrice']),\n            'ask': parseFloat (ticker['AskPrice']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['LastPrice']),\n            'change': parseFloat (ticker['Change']),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': parseFloat (ticker['Volume']),\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetMarketsummary (this.extend ({\n            'market_id': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetMarketsummaries (params);\n        let result = {};\n        for (let i = 0; i < tickers.length; i++) {\n            let ticker = this.parseTicker (tickers[i]);\n            let symbol = ticker['symbol'];\n            result[symbol] = ticker;\n        }\n        return result;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderbook (this.extend ({\n            'market_id': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'BuyOrders', 'SellOrders', 'Price', 'Quantity');\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + path;\n        if (api == 'public') {\n            params = this.urlencode (params);\n            if (params.length)\n                url += '?' + params;\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        let success = this.safeInteger (response, 'success');\n        if (success != 1) {\n            throw new ExchangeError (response['message']);\n        }\n        return response['result'];\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class coinfloor extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'coinfloor',\n            'name': 'coinfloor',\n            'rateLimit': 1000,\n            'countries': 'UK',\n            'has': {\n                'CORS': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28246081-623fc164-6a1c-11e7-913f-bac0d5576c90.jpg',\n                'api': 'https://webapi.coinfloor.co.uk:8090/bist',\n                'www': 'https://www.coinfloor.co.uk',\n                'doc': [\n                    'https://github.com/coinfloor/api',\n                    'https://www.coinfloor.co.uk/api',\n                ],\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'uid': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        '{id}/ticker/',\n                        '{id}/order_book/',\n                        '{id}/transactions/',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        '{id}/balance/',\n                        '{id}/user_transactions/',\n                        '{id}/open_orders/',\n                        '{id}/cancel_order/',\n                        '{id}/buy/',\n                        '{id}/sell/',\n                        '{id}/buy_market/',\n                        '{id}/sell_market/',\n                        '{id}/estimate_sell_market/',\n                        '{id}/estimate_buy_market/',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/GBP': { 'id': 'XBT/GBP', 'symbol': 'BTC/GBP', 'base': 'BTC', 'quote': 'GBP' },\n                'BTC/EUR': { 'id': 'XBT/EUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR' },\n                'BTC/USD': { 'id': 'XBT/USD', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD' },\n                'BTC/PLN': { 'id': 'XBT/PLN', 'symbol': 'BTC/PLN', 'base': 'BTC', 'quote': 'PLN' },\n                'BCH/GBP': { 'id': 'BCH/GBP', 'symbol': 'BCH/GBP', 'base': 'BCH', 'quote': 'GBP' },\n            },\n        });\n    }\n\n    fetchBalance (params = {}) {\n        let symbol = undefined;\n        if ('symbol' in params)\n            symbol = params['symbol'];\n        if ('id' in params)\n            symbol = params['id'];\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchBalance requires a symbol param');\n        // todo parse balance\n        return this.privatePostIdBalance ({\n            'id': this.marketId (symbol),\n        });\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetIdOrderBook (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        // rewrite to get the timestamp from HTTP headers\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let vwap = this.safeFloat (ticker, 'vwap');\n        let baseVolume = parseFloat (ticker['volume']);\n        let quoteVolume = undefined;\n        if (typeof vwap !== 'undefined') {\n            quoteVolume = baseVolume * vwap;\n        }\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': vwap,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let market = this.market (symbol);\n        let ticker = await this.publicGetIdTicker (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['date'] * 1000;\n        return {\n            'info': trade,\n            'id': trade['tid'].toString (),\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetIdTransactions (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let order = { 'id': this.marketId (symbol) };\n        let method = 'privatePostId' + this.capitalize (side);\n        if (type == 'market') {\n            order['quantity'] = amount;\n            method += 'Market';\n        } else {\n            order['price'] = price;\n            order['amount'] = amount;\n        }\n        return this[method] (this.extend (order, params));\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostIdCancelOrder ({ 'id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        // curl -k -u '[User ID]/[API key]:[Passphrase]' https://webapi.coinfloor.co.uk:8090/bist/XBT/GBP/balance/\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({ 'nonce': nonce }, query));\n            let auth = this.uid + '/' + this.apiKey + ':' + this.password;\n            let signature = this.stringToBase64 (auth);\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Authorization': 'Basic ' + signature,\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class coingi extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'coingi',\n            'name': 'Coingi',\n            'rateLimit': 1000,\n            'countries': [ 'PA', 'BG', 'CN', 'US' ], // Panama, Bulgaria, China, US\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28619707-5c9232a8-7212-11e7-86d6-98fe5d15cc6e.jpg',\n                'api': {\n                    'www': 'https://coingi.com',\n                    'current': 'https://api.coingi.com',\n                    'user': 'https://api.coingi.com',\n                },\n                'www': 'https://coingi.com',\n                'doc': 'http://docs.coingi.apiary.io/',\n            },\n            'api': {\n                'www': {\n                    'get': [\n                        '',\n                    ],\n                },\n                'current': {\n                    'get': [\n                        'order-book/{pair}/{askCount}/{bidCount}/{depth}',\n                        'transactions/{pair}/{maxCount}',\n                        '24hour-rolling-aggregation',\n                    ],\n                },\n                'user': {\n                    'post': [\n                        'balance',\n                        'add-order',\n                        'cancel-order',\n                        'orders',\n                        'transactions',\n                        'create-crypto-withdrawal',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'taker': 0.2 / 100,\n                    'maker': 0.2 / 100,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.001,\n                        'LTC': 0.01,\n                        'DOGE': 2,\n                        'PPC': 0.02,\n                        'VTC': 0.2,\n                        'NMC': 2,\n                        'DASH': 0.002,\n                        'USD': 10,\n                        'EUR': 10,\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                        'LTC': 0,\n                        'DOGE': 0,\n                        'PPC': 0,\n                        'VTC': 0,\n                        'NMC': 0,\n                        'DASH': 0,\n                        'USD': 5,\n                        'EUR': 1,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        this.parseJsonResponse = false;\n        let response = await this.wwwGet ();\n        this.parseJsonResponse = true;\n        let parts = response.split ('do=currencyPairSelector-selectCurrencyPair\" class=\"active\">');\n        let currencyParts = parts[1].split ('<div class=\"currency-pair-label\">');\n        let result = [];\n        for (let i = 1; i < currencyParts.length; i++) {\n            let currencyPart = currencyParts[i];\n            let idParts = currencyPart.split ('</div>');\n            let id = idParts[0];\n            let symbol = id;\n            id = id.replace ('/', '-');\n            id = id.toLowerCase ();\n            let [ base, quote ] = symbol.split ('/');\n            let precision = {\n                'amount': 8,\n                'price': 8,\n            };\n            let lot = Math.pow (10, -precision['amount']);\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': id,\n                'lot': lot,\n                'active': true,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': Math.pow (10, precision['amount']),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision['price']),\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': 0,\n                        'max': undefined,\n                    },\n                },\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let lowercaseCurrencies = [];\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            lowercaseCurrencies.push (currency.toLowerCase ());\n        }\n        let balances = await this.userPostBalance ({\n            'currencies': lowercaseCurrencies.join (','),\n        });\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency']['name'];\n            currency = currency.toUpperCase ();\n            let account = {\n                'free': balance['available'],\n                'used': balance['blocked'] + balance['inOrders'] + balance['withdrawing'],\n                'total': 0.0,\n            };\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orderbook = await this.currentGetOrderBookPairAskCountBidCountDepth (this.extend ({\n            'pair': market['id'],\n            'askCount': 512, // maximum returned number of asks 1-512\n            'bidCount': 512, // maximum returned number of bids 1-512\n            'depth': 32, // maximum number of depth range steps 1-32\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'baseAmount');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': ticker['high'],\n            'low': ticker['low'],\n            'bid': ticker['highestBid'],\n            'ask': ticker['lowestAsk'],\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': undefined,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': ticker['baseVolume'],\n            'quoteVolume': ticker['counterVolume'],\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.currentGet24hourRollingAggregation (params);\n        let result = {};\n        for (let t = 0; t < response.length; t++) {\n            let ticker = response[t];\n            let base = ticker['currencyPair']['base'].toUpperCase ();\n            let quote = ticker['currencyPair']['counter'].toUpperCase ();\n            let symbol = base + '/' + quote;\n            let market = undefined;\n            if (symbol in this.markets) {\n                market = this.markets[symbol];\n            }\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.fetchTickers (undefined, params);\n        if (symbol in tickers)\n            return tickers[symbol];\n        throw new ExchangeError (this.id + ' return did not contain ' + symbol);\n    }\n\n    parseTrade (trade, market = undefined) {\n        if (!market)\n            market = this.markets_by_id[trade['currencyPair']];\n        return {\n            'id': trade['id'],\n            'info': trade,\n            'timestamp': trade['timestamp'],\n            'datetime': this.iso8601 (trade['timestamp']),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined, // type\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.currentGetTransactionsPairMaxCount (this.extend ({\n            'pair': market['id'],\n            'maxCount': 128,\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'currencyPair': this.marketId (symbol),\n            'volume': amount,\n            'price': price,\n            'orderType': (side == 'buy') ? 0 : 1,\n        };\n        let response = await this.userPostAddOrder (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['result'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.userPostCancelOrder ({ 'orderId': id });\n    }\n\n    sign (path, api = 'current', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        if (api != 'www') {\n            url += '/' + api + '/' + this.implodeParams (path, params);\n        }\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'current') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else if (api == 'user') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let request = this.extend ({\n                'token': this.apiKey,\n                'nonce': nonce,\n            }, query);\n            let auth = nonce.toString () + '$' + this.apiKey;\n            request['signature'] = this.hmac (this.encode (auth), this.encode (this.secret));\n            body = this.json (request);\n            headers = {\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'current', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (typeof response !== 'string') {\n            if ('errors' in response)\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class coinmarketcap extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'coinmarketcap',\n            'name': 'CoinMarketCap',\n            'rateLimit': 10000,\n            'version': 'v1',\n            'countries': 'US',\n            'has': {\n                'CORS': true,\n                'privateAPI': false,\n                'createOrder': false,\n                'cancelOrder': false,\n                'fetchBalance': false,\n                'fetchOrderBook': false,\n                'fetchTrades': false,\n                'fetchTickers': true,\n                'fetchCurrencies': true,\n                'fetchCurrencies': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28244244-9be6312a-69ed-11e7-99c1-7c1797275265.jpg',\n                'api': {\n                    'public': 'https://api.coinmarketcap.com',\n                    'files': 'https://files.coinmarketcap.com',\n                    'charts': 'https://graph.coinmarketcap.com',\n                },\n                'www': 'https://coinmarketcap.com',\n                'doc': 'https://coinmarketcap.com/api',\n            },\n            'requiredCredentials': {\n                'apiKey': false,\n                'secret': false,\n            },\n            'api': {\n                'files': {\n                    'get': [\n                        'generated/stats/global.json',\n                    ],\n                },\n                'graphs': {\n                    'get': [\n                        'currencies/{name}/',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'ticker/',\n                        'ticker/{id}/',\n                        'global/',\n                    ],\n                },\n            },\n            'currencyCodes': [\n                'AUD',\n                'BRL',\n                'CAD',\n                'CHF',\n                'CNY',\n                'EUR',\n                'GBP',\n                'HKD',\n                'IDR',\n                'INR',\n                'JPY',\n                'KRW',\n                'MXN',\n                'RUB',\n                'USD',\n            ],\n        });\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        throw new ExchangeError ('Fetching order books is not supported by the API of ' + this.id);\n    }\n\n    currencyCode (base, name) {\n        const currencies = {\n            'Bitgem': 'Bitgem',\n            'NetCoin': 'NetCoin',\n            'BatCoin': 'BatCoin',\n        };\n        if (name in currencies)\n            return currencies[name];\n        return base;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetTicker ({\n            'limit': 0,\n        });\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let currencies = this.currencyCodes;\n            for (let i = 0; i < currencies.length; i++) {\n                let quote = currencies[i];\n                let quoteId = quote.toLowerCase ();\n                let baseId = market['id'];\n                let base = this.currencyCode (market['symbol'], market['name']);\n                let symbol = base + '/' + quote;\n                let id = baseId + '/' + quote;\n                result.push ({\n                    'id': id,\n                    'symbol': symbol,\n                    'base': base,\n                    'quote': quote,\n                    'baseId': baseId,\n                    'quoteId': quoteId,\n                    'info': market,\n                });\n            }\n        }\n        return result;\n    }\n\n    async fetchGlobal (currency = 'USD') {\n        await this.loadMarkets ();\n        let request = {};\n        if (currency)\n            request['convert'] = currency;\n        return await this.publicGetGlobal (request);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        if ('last_updated' in ticker)\n            if (ticker['last_updated'])\n                timestamp = parseInt (ticker['last_updated']) * 1000;\n        let change = undefined;\n        if ('percent_change_24h' in ticker)\n            if (ticker['percent_change_24h'])\n                change = this.safeFloat (ticker, 'percent_change_24h');\n        let last = undefined;\n        let symbol = undefined;\n        let volume = undefined;\n        if (market) {\n            let priceKey = 'price_' + market['quoteId'];\n            if (priceKey in ticker)\n                if (ticker[priceKey])\n                    last = this.safeFloat (ticker, priceKey);\n            symbol = market['symbol'];\n            let volumeKey = '24h_volume_' + market['quoteId'];\n            if (volumeKey in ticker)\n                if (ticker[volumeKey])\n                    volume = this.safeFloat (ticker, volumeKey);\n        }\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': undefined,\n            'ask': undefined,\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': last,\n            'change': change,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': volume,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (currency = 'USD', params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            'limit': 10000,\n        };\n        if (currency)\n            request['convert'] = currency;\n        let response = await this.publicGetTicker (this.extend (request, params));\n        let tickers = {};\n        for (let t = 0; t < response.length; t++) {\n            let ticker = response[t];\n            let id = ticker['id'] + '/' + currency;\n            let symbol = id;\n            let market = undefined;\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            }\n            tickers[symbol] = this.parseTicker (ticker, market);\n        }\n        return tickers;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = this.extend ({\n            'convert': market['quote'],\n            'id': market['baseId'],\n        }, params);\n        let response = await this.publicGetTickerId (request);\n        let ticker = response[0];\n        return this.parseTicker (ticker, market);\n    }\n\n    async fetchCurrencies (params = {}) {\n        let currencies = await this.publicGetTicker (this.extend ({\n            'limit': 0,\n        }, params));\n        let result = {};\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let id = currency['symbol'];\n            let name = currency['name'];\n            // todo: will need to rethink the fees\n            // to add support for multiple withdrawal/deposit methods and\n            // differentiated fees for each particular method\n            let precision = 8; // default precision, todo: fix \"magic constants\"\n            let code = this.currencyCode (id, name);\n            result[code] = {\n                'id': id,\n                'code': code,\n                'info': currency,\n                'name': name,\n                'active': true,\n                'status': 'ok',\n                'fee': undefined, // todo: redesign\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                },\n            };\n        }\n        return result;\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api] + '/' + this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (Object.keys (query).length)\n            url += '?' + this.urlencode (query);\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response) {\n            if (response['error']) {\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n            }\n        }\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class coinmate extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'coinmate',\n            'name': 'CoinMate',\n            'countries': [ 'GB', 'CZ', 'EU' ], // UK, Czech Republic\n            'rateLimit': 1000,\n            'has': {\n                'CORS': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27811229-c1efb510-606c-11e7-9a36-84ba2ce412d8.jpg',\n                'api': 'https://coinmate.io/api',\n                'www': 'https://coinmate.io',\n                'doc': [\n                    'http://docs.coinmate.apiary.io',\n                    'https://coinmate.io/developers',\n                ],\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'uid': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'orderBook',\n                        'ticker',\n                        'transactions',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balances',\n                        'bitcoinWithdrawal',\n                        'bitcoinDepositAddresses',\n                        'buyInstant',\n                        'buyLimit',\n                        'cancelOrder',\n                        'cancelOrderWithInfo',\n                        'createVoucher',\n                        'openOrders',\n                        'redeemVoucher',\n                        'sellInstant',\n                        'sellLimit',\n                        'transactionHistory',\n                        'unconfirmedBitcoinDeposits',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/EUR': { 'id': 'BTC_EUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR', 'precision': { 'amount': 4, 'price': 2 }},\n                'BTC/CZK': { 'id': 'BTC_CZK', 'symbol': 'BTC/CZK', 'base': 'BTC', 'quote': 'CZK', 'precision': { 'amount': 4, 'price': 2 }},\n                'LTC/BTC': { 'id': 'LTC_BTC', 'symbol': 'LTC/BTC', 'base': 'LTC', 'quote': 'BTC', 'precision': { 'amount': 4, 'price': 5 }},\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.0005,\n                    'taker': 0.0035,\n                },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostBalances ();\n        let balances = response['data'];\n        let result = { 'info': balances };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            if (currency in balances) {\n                account['free'] = balances[currency]['available'];\n                account['used'] = balances[currency]['reserved'];\n                account['total'] = balances[currency]['balance'];\n            }\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let response = await this.publicGetOrderBook (this.extend ({\n            'currencyPair': this.marketId (symbol),\n            'groupByPriceLimit': 'False',\n        }, params));\n        let orderbook = response['data'];\n        let timestamp = orderbook['timestamp'] * 1000;\n        return this.parseOrderBook (orderbook, timestamp, 'bids', 'asks', 'price', 'amount');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let response = await this.publicGetTicker (this.extend ({\n            'currencyPair': this.marketId (symbol),\n        }, params));\n        let ticker = response['data'];\n        let timestamp = ticker['timestamp'] * 1000;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['amount']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        if (!market)\n            market = this.markets_by_id[trade['currencyPair']];\n        return {\n            'id': trade['transactionId'],\n            'info': trade,\n            'timestamp': trade['timestamp'],\n            'datetime': this.iso8601 (trade['timestamp']),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetTransactions (this.extend ({\n            'currencyPair': market['id'],\n            'minutesIntoHistory': 10,\n        }, params));\n        return this.parseTrades (response['data'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let method = 'privatePost' + this.capitalize (side);\n        let order = {\n            'currencyPair': this.marketId (symbol),\n        };\n        if (type == 'market') {\n            if (side == 'buy')\n                order['total'] = amount; // amount in fiat\n            else\n                order['amount'] = amount; // amount in fiat\n            method += 'Instant';\n        } else {\n            order['amount'] = amount; // amount in crypto\n            order['price'] = price;\n            method += this.capitalize (type);\n        }\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['data'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelOrder ({ 'orderId': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + path;\n        if (api == 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let auth = nonce + this.uid + this.apiKey;\n            let signature = this.hmac (this.encode (auth), this.encode (this.secret));\n            body = this.urlencode (this.extend ({\n                'clientId': this.uid,\n                'nonce': nonce,\n                'publicKey': this.apiKey,\n                'signature': signature.toUpperCase (),\n            }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response)\n            if (response['error'])\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class coinsecure extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'coinsecure',\n            'name': 'Coinsecure',\n            'countries': 'IN', // India\n            'rateLimit': 1000,\n            'version': 'v1',\n            'has': {\n                'CORS': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766472-9cbd200a-5ed9-11e7-9551-2267ad7bac08.jpg',\n                'api': 'https://api.coinsecure.in',\n                'www': 'https://coinsecure.in',\n                'doc': [\n                    'https://api.coinsecure.in',\n                    'https://github.com/coinsecure/plugins',\n                ],\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': false,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'bitcoin/search/confirmation/{txid}',\n                        'exchange/ask/low',\n                        'exchange/ask/orders',\n                        'exchange/bid/high',\n                        'exchange/bid/orders',\n                        'exchange/lastTrade',\n                        'exchange/max24Hr',\n                        'exchange/min24Hr',\n                        'exchange/ticker',\n                        'exchange/trades',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'mfa/authy/call',\n                        'mfa/authy/sms',\n                        'netki/search/{netkiName}',\n                        'user/bank/otp/{number}',\n                        'user/kyc/otp/{number}',\n                        'user/profile/phone/otp/{number}',\n                        'user/wallet/coin/address/{id}',\n                        'user/wallet/coin/deposit/confirmed/all',\n                        'user/wallet/coin/deposit/confirmed/{id}',\n                        'user/wallet/coin/deposit/unconfirmed/all',\n                        'user/wallet/coin/deposit/unconfirmed/{id}',\n                        'user/wallet/coin/wallets',\n                        'user/exchange/bank/fiat/accounts',\n                        'user/exchange/bank/fiat/balance/available',\n                        'user/exchange/bank/fiat/balance/pending',\n                        'user/exchange/bank/fiat/balance/total',\n                        'user/exchange/bank/fiat/deposit/cancelled',\n                        'user/exchange/bank/fiat/deposit/unverified',\n                        'user/exchange/bank/fiat/deposit/verified',\n                        'user/exchange/bank/fiat/withdraw/cancelled',\n                        'user/exchange/bank/fiat/withdraw/completed',\n                        'user/exchange/bank/fiat/withdraw/unverified',\n                        'user/exchange/bank/fiat/withdraw/verified',\n                        'user/exchange/ask/cancelled',\n                        'user/exchange/ask/completed',\n                        'user/exchange/ask/pending',\n                        'user/exchange/bid/cancelled',\n                        'user/exchange/bid/completed',\n                        'user/exchange/bid/pending',\n                        'user/exchange/bank/coin/addresses',\n                        'user/exchange/bank/coin/balance/available',\n                        'user/exchange/bank/coin/balance/pending',\n                        'user/exchange/bank/coin/balance/total',\n                        'user/exchange/bank/coin/deposit/cancelled',\n                        'user/exchange/bank/coin/deposit/unverified',\n                        'user/exchange/bank/coin/deposit/verified',\n                        'user/exchange/bank/coin/withdraw/cancelled',\n                        'user/exchange/bank/coin/withdraw/completed',\n                        'user/exchange/bank/coin/withdraw/unverified',\n                        'user/exchange/bank/coin/withdraw/verified',\n                        'user/exchange/bank/summary',\n                        'user/exchange/coin/fee',\n                        'user/exchange/fiat/fee',\n                        'user/exchange/kycs',\n                        'user/exchange/referral/coin/paid',\n                        'user/exchange/referral/coin/successful',\n                        'user/exchange/referral/fiat/paid',\n                        'user/exchange/referrals',\n                        'user/exchange/trade/summary',\n                        'user/login/token/{token}',\n                        'user/summary',\n                        'user/wallet/summary',\n                        'wallet/coin/withdraw/cancelled',\n                        'wallet/coin/withdraw/completed',\n                        'wallet/coin/withdraw/unverified',\n                        'wallet/coin/withdraw/verified',\n                    ],\n                    'post': [\n                        'login',\n                        'login/initiate',\n                        'login/password/forgot',\n                        'mfa/authy/initiate',\n                        'mfa/ga/initiate',\n                        'signup',\n                        'user/netki/update',\n                        'user/profile/image/update',\n                        'user/exchange/bank/coin/withdraw/initiate',\n                        'user/exchange/bank/coin/withdraw/newVerifycode',\n                        'user/exchange/bank/fiat/withdraw/initiate',\n                        'user/exchange/bank/fiat/withdraw/newVerifycode',\n                        'user/password/change',\n                        'user/password/reset',\n                        'user/wallet/coin/withdraw/initiate',\n                        'wallet/coin/withdraw/newVerifycode',\n                    ],\n                    'put': [\n                        'signup/verify/{token}',\n                        'user/exchange/kyc',\n                        'user/exchange/bank/fiat/deposit/new',\n                        'user/exchange/ask/new',\n                        'user/exchange/bid/new',\n                        'user/exchange/instant/buy',\n                        'user/exchange/instant/sell',\n                        'user/exchange/bank/coin/withdraw/verify',\n                        'user/exchange/bank/fiat/account/new',\n                        'user/exchange/bank/fiat/withdraw/verify',\n                        'user/mfa/authy/initiate/enable',\n                        'user/mfa/ga/initiate/enable',\n                        'user/netki/create',\n                        'user/profile/phone/new',\n                        'user/wallet/coin/address/new',\n                        'user/wallet/coin/new',\n                        'user/wallet/coin/withdraw/sendToExchange',\n                        'user/wallet/coin/withdraw/verify',\n                    ],\n                    'delete': [\n                        'user/gcm/{code}',\n                        'user/logout',\n                        'user/exchange/bank/coin/withdraw/unverified/cancel/{withdrawID}',\n                        'user/exchange/bank/fiat/deposit/cancel/{depositID}',\n                        'user/exchange/ask/cancel/{orderID}',\n                        'user/exchange/bid/cancel/{orderID}',\n                        'user/exchange/bank/fiat/withdraw/unverified/cancel/{withdrawID}',\n                        'user/mfa/authy/disable/{code}',\n                        'user/mfa/ga/disable/{code}',\n                        'user/profile/phone/delete',\n                        'user/profile/image/delete/{netkiName}',\n                        'user/wallet/coin/withdraw/unverified/cancel/{withdrawID}',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/INR': { 'id': 'BTC/INR', 'symbol': 'BTC/INR', 'base': 'BTC', 'quote': 'INR' },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.4 / 100,\n                    'taker': 0.4 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privateGetUserExchangeBankSummary ();\n        let balance = response['message'];\n        let coin = {\n            'free': balance['availableCoinBalance'],\n            'used': balance['pendingCoinBalance'],\n            'total': balance['totalCoinBalance'],\n        };\n        let fiat = {\n            'free': balance['availableFiatBalance'],\n            'used': balance['pendingFiatBalance'],\n            'total': balance['totalFiatBalance'],\n        };\n        let result = {\n            'info': balance,\n            'BTC': coin,\n            'INR': fiat,\n        };\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let bids = await this.publicGetExchangeBidOrders (params);\n        let asks = await this.publicGetExchangeAskOrders (params);\n        let orderbook = {\n            'bids': bids['message'],\n            'asks': asks['message'],\n        };\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'rate', 'vol');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let response = await this.publicGetExchangeTicker (params);\n        let ticker = response['message'];\n        let timestamp = ticker['timestamp'];\n        let baseVolume = parseFloat (ticker['coinvolume']);\n        if (symbol == 'BTC/INR') {\n            let satoshi = 0.00000001;\n            baseVolume = baseVolume * satoshi;\n        }\n        let quoteVolume = parseFloat (ticker['fiatvolume']) / 100;\n        let vwap = quoteVolume / baseVolume;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']) / 100,\n            'low': parseFloat (ticker['low']) / 100,\n            'bid': parseFloat (ticker['bid']) / 100,\n            'ask': parseFloat (ticker['ask']) / 100,\n            'vwap': vwap,\n            'open': parseFloat (ticker['open']) / 100,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['lastPrice']) / 100,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, symbol = undefined) {\n        let timestamp = trade['time'];\n        let side = (trade['ordType'] == 'bid') ? 'buy' : 'sell';\n        return {\n            'id': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'order': undefined,\n            'symbol': symbol,\n            'type': undefined,\n            'side': side,\n            'price': this.safeFloat (trade, 'rate') / 100,\n            'amount': this.safeFloat (trade, 'vol') / 100000000,\n            'fee': undefined,\n            'info': trade,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let result = await this.publicGetExchangeTrades (params);\n        if ('message' in result) {\n            let trades = result['message'];\n            return this.parseTrades (trades, symbol);\n        }\n    }\n\n    async createOrder (market, type, side, amount, price = undefined, params = {}) {\n        let method = 'privatePutUserExchange';\n        let order = {};\n        if (type == 'market') {\n            method += 'Instant' + this.capitalize (side);\n            if (side == 'buy')\n                order['maxFiat'] = amount;\n            else\n                order['maxVol'] = amount;\n        } else {\n            let direction = (side == 'buy') ? 'Bid' : 'Ask';\n            method += direction + 'New';\n            order['rate'] = price;\n            order['vol'] = amount;\n        }\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['message']['orderID'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        throw new ExchangeError (this.id + ' cancelOrder () is not fully implemented yet');\n        let method = 'privateDeleteUserExchangeAskCancelOrderId'; // TODO fixme, have to specify order side here\n        return await this[method] ({ 'orderID': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            headers = { 'Authorization': this.apiKey };\n            if (Object.keys (query).length) {\n                body = this.json (query);\n                headers['Content-Type'] = 'application/json';\n            }\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code == 200) {\n            if ((body[0] == '{') || (body[0] == '[')) {\n                let response = JSON.parse (body);\n                if ('success' in response) {\n                    let success = response['success'];\n                    if (!success) {\n                        throw new ExchangeError (this.id + ' error returned: ' + body);\n                    }\n                    if (!('message' in response)) {\n                        throw new ExchangeError (this.id + ' malformed response: no \"message\" in response: ' + body);\n                    }\n                } else {\n                    throw new ExchangeError (this.id + ' malformed response: no \"success\" in response: ' + body);\n                }\n            } else {\n                // if not a JSON response\n                throw new ExchangeError (this.id + ' returned a non-JSON reply: ' + body);\n            }\n        }\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, AuthenticationError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class coinspot extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'coinspot',\n            'name': 'CoinSpot',\n            'countries': 'AU', // Australia\n            'rateLimit': 1000,\n            'has': {\n                'CORS': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28208429-3cacdf9a-6896-11e7-854e-4c79a772a30f.jpg',\n                'api': {\n                    'public': 'https://www.coinspot.com.au/pubapi',\n                    'private': 'https://www.coinspot.com.au/api',\n                },\n                'www': 'https://www.coinspot.com.au',\n                'doc': 'https://www.coinspot.com.au/api',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'latest',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'orders',\n                        'orders/history',\n                        'my/coin/deposit',\n                        'my/coin/send',\n                        'quote/buy',\n                        'quote/sell',\n                        'my/balances',\n                        'my/orders',\n                        'my/buy',\n                        'my/sell',\n                        'my/buy/cancel',\n                        'my/sell/cancel',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/AUD': { 'id': 'BTC', 'symbol': 'BTC/AUD', 'base': 'BTC', 'quote': 'AUD' },\n                'LTC/AUD': { 'id': 'LTC', 'symbol': 'LTC/AUD', 'base': 'LTC', 'quote': 'AUD' },\n                'DOGE/AUD': { 'id': 'DOGE', 'symbol': 'DOGE/AUD', 'base': 'DOGE', 'quote': 'AUD' },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostMyBalances ();\n        let result = { 'info': response };\n        if ('balance' in response) {\n            let balances = response['balance'];\n            let currencies = Object.keys (balances);\n            for (let c = 0; c < currencies.length; c++) {\n                let currency = currencies[c];\n                let uppercase = currency.toUpperCase ();\n                let account = {\n                    'free': balances[currency],\n                    'used': 0.0,\n                    'total': balances[currency],\n                };\n                if (uppercase == 'DRK')\n                    uppercase = 'DASH';\n                result[uppercase] = account;\n            }\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let market = this.market (symbol);\n        let orderbook = await this.privatePostOrders (this.extend ({\n            'cointype': market['id'],\n        }, params));\n        let result = this.parseOrderBook (orderbook, undefined, 'buyorders', 'sellorders', 'rate', 'amount');\n        result['bids'] = this.sortBy (result['bids'], 0, true);\n        result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let response = await this.publicGetLatest (params);\n        let id = this.marketId (symbol);\n        id = id.toLowerCase ();\n        let ticker = response['prices'][id];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        return this.privatePostOrdersHistory (this.extend ({\n            'cointype': this.marketId (symbol),\n        }, params));\n    }\n\n    createOrder (market, type, side, amount, price = undefined, params = {}) {\n        let method = 'privatePostMy' + this.capitalize (side);\n        if (type == 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let order = {\n            'cointype': this.marketId (market),\n            'amount': amount,\n            'rate': price,\n        };\n        return this[method] (this.extend (order, params));\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        throw new ExchangeError (this.id + ' cancelOrder () is not fully implemented yet');\n        let method = 'privatePostMyBuy';\n        return await this[method] ({ 'id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        if (!this.apiKey)\n            throw new AuthenticationError (this.id + ' requires apiKey for all requests');\n        let url = this.urls['api'][api] + '/' + path;\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.json (this.extend ({ 'nonce': nonce }, params));\n            headers = {\n                'Content-Type': 'application/json',\n                'key': this.apiKey,\n                'sign': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InsufficientFunds, OrderNotFound, OrderNotCached } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class cryptopia extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'cryptopia',\n            'name': 'Cryptopia',\n            'rateLimit': 1500,\n            'countries': 'NZ', // New Zealand\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n                'fetchOrder': 'emulated',\n                'fetchOrders': 'emulated',\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': 'emulated',\n                'fetchMyTrades': true,\n                'fetchCurrencies': true,\n                'deposit': true,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/29484394-7b4ea6e2-84c6-11e7-83e5-1fccf4b2dc81.jpg',\n                'api': 'https://www.cryptopia.co.nz/api',\n                'www': 'https://www.cryptopia.co.nz',\n                'doc': [\n                    'https://www.cryptopia.co.nz/Forum/Category/45',\n                    'https://www.cryptopia.co.nz/Forum/Thread/255',\n                    'https://www.cryptopia.co.nz/Forum/Thread/256',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'GetCurrencies',\n                        'GetTradePairs',\n                        'GetMarkets',\n                        'GetMarkets/{id}',\n                        'GetMarkets/{hours}',\n                        'GetMarkets/{id}/{hours}',\n                        'GetMarket/{id}',\n                        'GetMarket/{id}/{hours}',\n                        'GetMarketHistory/{id}',\n                        'GetMarketHistory/{id}/{hours}',\n                        'GetMarketOrders/{id}',\n                        'GetMarketOrders/{id}/{count}',\n                        'GetMarketOrderGroups/{ids}/{count}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'CancelTrade',\n                        'GetBalance',\n                        'GetDepositAddress',\n                        'GetOpenOrders',\n                        'GetTradeHistory',\n                        'GetTransactions',\n                        'SubmitTip',\n                        'SubmitTrade',\n                        'SubmitTransfer',\n                        'SubmitWithdraw',\n                    ],\n                },\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        const currencies = {\n            'ACC': 'AdCoin',\n            'CC': 'CCX',\n            'CMT': 'Comet',\n            'FCN': 'Facilecoin',\n            'NET': 'NetCoin',\n            'BTG': 'Bitgem',\n            'FUEL': 'FC2', // FuelCoin != FUEL\n            'QBT': 'Cubits',\n            'WRC': 'WarCoin',\n        };\n        if (currency in currencies)\n            return currencies[currency];\n        return currency;\n    }\n\n    currencyId (currency) {\n        const currencies = {\n            'AdCoin': 'ACC',\n            'CCX': 'CC',\n            'Comet': 'CMT',\n            'Cubits': 'QBT',\n            'Facilecoin': 'FCN',\n            'NetCoin': 'NET',\n            'Bitgem': 'BTG',\n            'FC2': 'FUEL',\n        };\n        if (currency in currencies)\n            return currencies[currency];\n        return currency;\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicGetTradePairs ();\n        let result = [];\n        let markets = response['Data'];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let id = market['Id'];\n            let symbol = market['Label'];\n            let [ base, quote ] = symbol.split ('/');\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            symbol = base + '/' + quote;\n            let precision = {\n                'amount': 8,\n                'price': 8,\n            };\n            let lot = market['MinimumTrade'];\n            let priceLimits = {\n                'min': market['MinimumPrice'],\n                'max': market['MaximumPrice'],\n            };\n            let amountLimits = {\n                'min': lot,\n                'max': market['MaximumTrade'],\n            };\n            let limits = {\n                'amount': amountLimits,\n                'price': priceLimits,\n                'cost': {\n                    'min': priceLimits['min'] * amountLimits['min'],\n                    'max': undefined,\n                },\n            };\n            let active = market['Status'] === 'OK';\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n                'maker': market['TradeFee'] / 100,\n                'taker': market['TradeFee'] / 100,\n                'lot': limits['amount']['min'],\n                'active': active,\n                'precision': precision,\n                'limits': limits,\n            });\n        }\n        return result;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetMarketOrdersId (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        let orderbook = response['Data'];\n        return this.parseOrderBook (orderbook, undefined, 'Buy', 'Sell', 'Price', 'Volume');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'info': ticker,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['High']),\n            'low': parseFloat (ticker['Low']),\n            'bid': parseFloat (ticker['BidPrice']),\n            'ask': parseFloat (ticker['AskPrice']),\n            'vwap': undefined,\n            'open': parseFloat (ticker['Open']),\n            'close': parseFloat (ticker['Close']),\n            'first': undefined,\n            'last': parseFloat (ticker['LastPrice']),\n            'change': parseFloat (ticker['Change']),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['Volume']),\n            'quoteVolume': parseFloat (ticker['BaseVolume']),\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetMarketId (this.extend ({\n            'id': market['id'],\n        }, params));\n        let ticker = response['Data'];\n        return this.parseTicker (ticker, market);\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetMarkets (params);\n        let result = {};\n        let tickers = response['Data'];\n        for (let i = 0; i < tickers.length; i++) {\n            let ticker = tickers[i];\n            let id = ticker['TradePairId'];\n            let recognized = (id in this.markets_by_id);\n            if (!recognized)\n                throw new ExchangeError (this.id + ' fetchTickers() returned unrecognized pair id ' + id);\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = undefined;\n        if ('Timestamp' in trade) {\n            timestamp = trade['Timestamp'] * 1000;\n        } else if ('TimeStamp' in trade) {\n            timestamp = this.parse8601 (trade['TimeStamp']);\n        }\n        let price = this.safeFloat (trade, 'Price');\n        if (!price)\n            price = this.safeFloat (trade, 'Rate');\n        let cost = this.safeFloat (trade, 'Total');\n        let id = this.safeString (trade, 'TradeId');\n        if (!market) {\n            if ('TradePairId' in trade)\n                if (trade['TradePairId'] in this.markets_by_id)\n                    market = this.markets_by_id[trade['TradePairId']];\n        }\n        let symbol = undefined;\n        let fee = undefined;\n        if (market) {\n            symbol = market['symbol'];\n            if ('Fee' in trade) {\n                fee = {\n                    'currency': market['quote'],\n                    'cost': trade['Fee'],\n                };\n            }\n        }\n        return {\n            'id': id,\n            'info': trade,\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': 'limit',\n            'side': trade['Type'].toLowerCase (),\n            'price': price,\n            'cost': cost,\n            'amount': trade['Amount'],\n            'fee': fee,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let hours = 24; // the default\n        if (since) {\n            let elapsed = this.milliseconds () - since;\n            let hour = 1000 * 60 * 60;\n            hours = parseInt (elapsed / hour);\n        }\n        let request = {\n            'id': market['id'],\n            'hours': hours,\n        };\n        let response = await this.publicGetMarketHistoryIdHours (this.extend (request, params));\n        let trades = response['Data'];\n        return this.parseTrades (trades, market, since, limit);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['TradePairId'] = market['id'];\n        }\n        let response = await this.privatePostGetTradeHistory (this.extend (request, params));\n        return this.parseTrades (response['Data'], market, since, limit);\n    }\n\n    async fetchCurrencies (params = {}) {\n        let response = await this.publicGetCurrencies (params);\n        let currencies = response['Data'];\n        let result = {};\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let id = currency['Symbol'];\n            // todo: will need to rethink the fees\n            // to add support for multiple withdrawal/deposit methods and\n            // differentiated fees for each particular method\n            let precision = 8; // default precision, todo: fix \"magic constants\"\n            let code = this.commonCurrencyCode (id);\n            let active = (currency['ListingStatus'] === 'Active');\n            let status = currency['Status'].toLowerCase ();\n            if (status !== 'ok')\n                active = false;\n            result[code] = {\n                'id': id,\n                'code': code,\n                'info': currency,\n                'name': currency['Name'],\n                'active': active,\n                'status': status,\n                'fee': currency['WithdrawFee'],\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': currency['MinBaseTrade'],\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': currency['MinWithdraw'],\n                        'max': currency['MaxWithdraw'],\n                    },\n                },\n            };\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetBalance ();\n        let balances = response['Data'];\n        let result = { 'info': response };\n        for (let i = 0; i < balances.length; i++) {\n            let balance = balances[i];\n            let code = balance['Symbol'];\n            let currency = this.commonCurrencyCode (code);\n            let account = {\n                'free': balance['Available'],\n                'used': 0.0,\n                'total': balance['Total'],\n            };\n            account['used'] = account['total'] - account['free'];\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type === 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        // price = parseFloat (price);\n        // amount = parseFloat (amount);\n        let request = {\n            'TradePairId': market['id'],\n            'Type': this.capitalize (side),\n            // 'Rate': this.priceToPrecision (symbol, price),\n            // 'Amount': this.amountToPrecision (symbol, amount),\n            'Rate': price,\n            'Amount': amount,\n        };\n        let response = await this.privatePostSubmitTrade (this.extend (request, params));\n        if (!response)\n            throw new ExchangeError (this.id + ' createOrder returned unknown error: ' + this.json (response));\n        let id = undefined;\n        let filled = 0.0;\n        if ('Data' in response) {\n            if ('OrderId' in response['Data']) {\n                if (response['Data']['OrderId']) {\n                    id = response['Data']['OrderId'].toString ();\n                }\n            }\n            if ('FilledOrders' in response['Data']) {\n                let filledOrders = response['Data']['FilledOrders'];\n                let filledOrdersLength = filledOrders.length;\n                if (filledOrdersLength) {\n                    filled = undefined;\n                }\n            }\n        }\n        let timestamp = this.milliseconds ();\n        let order = {\n            'id': id,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': 'open',\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': price,\n            'cost': price * amount,\n            'amount': amount,\n            'remaining': amount,\n            'filled': filled,\n            'fee': undefined,\n            // 'trades': this.parseTrades (order['trades'], market),\n        };\n        if (id)\n            this.orders[id] = order;\n        return this.extend ({ 'info': response }, order);\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = undefined;\n        try {\n            response = await this.privatePostCancelTrade (this.extend ({\n                'Type': 'Trade',\n                'OrderId': id,\n            }, params));\n            if (id in this.orders)\n                this.orders[id]['status'] = 'canceled';\n        } catch (e) {\n            if (this.last_json_response) {\n                let message = this.safeString (this.last_json_response, 'Error');\n                if (message) {\n                    if (message.indexOf ('does not exist') >= 0)\n                        throw new OrderNotFound (this.id + ' cancelOrder() error: ' + this.last_http_response);\n                }\n            }\n            throw e;\n        }\n        return response;\n    }\n\n    parseOrder (order, market = undefined) {\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else if ('Market' in order) {\n            let id = order['Market'];\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            }\n        }\n        let timestamp = this.parse8601 (order['TimeStamp']);\n        let amount = this.safeFloat (order, 'Amount');\n        let remaining = this.safeFloat (order, 'Remaining');\n        let filled = amount - remaining;\n        return {\n            'id': order['OrderId'].toString (),\n            'info': this.omit (order, 'status'),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': order['status'],\n            'symbol': symbol,\n            'type': 'limit',\n            'side': order['Type'].toLowerCase (),\n            'price': this.safeFloat (order, 'Rate'),\n            'cost': this.safeFloat (order, 'Total'),\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'fee': undefined,\n            // 'trades': this.parseTrades (order['trades'], market),\n        };\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrders requires a symbol param');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.privatePostGetOpenOrders ({\n            // 'Market': market['id'],\n            'TradePairId': market['id'], // Cryptopia identifier (not required if 'Market' supplied)\n            // 'Count': 100, // default = 100\n        }, params);\n        let orders = [];\n        for (let i = 0; i < response['Data'].length; i++) {\n            orders.push (this.extend (response['Data'][i], { 'status': 'open' }));\n        }\n        let openOrders = this.parseOrders (orders, market);\n        for (let j = 0; j < openOrders.length; j++) {\n            this.orders[openOrders[j]['id']] = openOrders[j];\n        }\n        let openOrdersIndexedById = this.indexBy (openOrders, 'id');\n        let cachedOrderIds = Object.keys (this.orders);\n        let result = [];\n        for (let k = 0; k < cachedOrderIds.length; k++) {\n            let id = cachedOrderIds[k];\n            if (id in openOrdersIndexedById) {\n                this.orders[id] = this.extend (this.orders[id], openOrdersIndexedById[id]);\n            } else {\n                let order = this.orders[id];\n                if (order['status'] === 'open') {\n                    this.orders[id] = this.extend (order, {\n                        'status': 'closed',\n                        'cost': order['amount'] * order['price'],\n                        'filled': order['amount'],\n                        'remaining': 0.0,\n                    });\n                }\n            }\n            let order = this.orders[id];\n            if (order['symbol'] === symbol)\n                result.push (order);\n        }\n        return this.filterBySinceLimit (result, since, limit);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        id = id.toString ();\n        let orders = await this.fetchOrders (symbol, undefined, undefined, params);\n        for (let i = 0; i < orders.length; i++) {\n            if (orders[i]['id'] === id)\n                return orders[i];\n        }\n        throw new OrderNotCached (this.id + ' order ' + id + ' not found in cached .orders, fetchOrder requires .orders (de)serialization implemented for this method to work properly');\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, params);\n        let result = [];\n        for (let i = 0; i < orders.length; i++) {\n            if (orders[i]['status'] === 'open')\n                result.push (orders[i]);\n        }\n        return result;\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, params);\n        let result = [];\n        for (let i = 0; i < orders.length; i++) {\n            if (orders[i]['status'] === 'closed')\n                result.push (orders[i]);\n        }\n        return result;\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        let currencyId = this.currencyId (currency);\n        let response = await this.privatePostGetDepositAddress (this.extend ({\n            'Currency': currencyId,\n        }, params));\n        let address = this.safeString (response['Data'], 'BaseAddress');\n        if (!address)\n            address = this.safeString (response['Data'], 'Address');\n        return {\n            'currency': currency,\n            'address': address,\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        let currencyId = this.currencyId (currency);\n        let request = {\n            'Currency': currencyId,\n            'Amount': amount,\n            'Address': address, // Address must exist in you AddressBook in security settings\n        };\n        if (tag)\n            request['PaymentId'] = tag;\n        let response = await this.privatePostSubmitWithdraw (this.extend (request, params));\n        return {\n            'info': response,\n            'id': response['Data'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            body = this.json (query);\n            let hash = this.hash (this.encode (body), 'md5', 'base64');\n            let secret = this.base64ToBinary (this.secret);\n            let uri = this.encodeURIComponent (url);\n            let lowercase = uri.toLowerCase ();\n            let payload = this.apiKey + method + lowercase + nonce + this.binaryToString (hash);\n            let signature = this.hmac (this.encode (payload), secret, 'sha256', 'base64');\n            let auth = 'amx ' + this.apiKey + ':' + this.binaryToString (signature) + ':' + nonce;\n            headers = {\n                'Content-Type': 'application/json',\n                'Authorization': auth,\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (response) {\n            if ('Success' in response)\n                if (response['Success']) {\n                    return response;\n                } else if ('Error' in response) {\n                    if (response['Error'] === 'Insufficient Funds.')\n                        throw new InsufficientFunds (this.id + ' ' + this.json (response));\n                }\n        }\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst liqui = require ('./liqui.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class dsx extends liqui {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'dsx',\n            'name': 'DSX',\n            'countries': 'UK',\n            'rateLimit': 1500,\n            'has': {\n                'CORS': false,\n                'fetchOrder': true,\n                'fetchOrders': true,\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': true,\n                'fetchTickers': true,\n                'fetchMyTrades': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27990275-1413158a-645a-11e7-931c-94717f7510e3.jpg',\n                'api': {\n                    'public': 'https://dsx.uk/mapi', // market data\n                    'private': 'https://dsx.uk/tapi', // trading\n                    'dwapi': 'https://dsx.uk/dwapi', // deposit/withdraw\n                },\n                'www': 'https://dsx.uk',\n                'doc': [\n                    'https://api.dsx.uk',\n                    'https://dsx.uk/api_docs/public',\n                    'https://dsx.uk/api_docs/private',\n                    '',\n                ],\n            },\n            'api': {\n                // market data (public)\n                'public': {\n                    'get': [\n                        'barsFromMoment/{id}/{period}/{start}', // empty reply :\\\n                        'depth/{pair}',\n                        'info',\n                        'lastBars/{id}/{period}/{amount}', // period is (m, h or d)\n                        'periodBars/{id}/{period}/{start}/{end}',\n                        'ticker/{pair}',\n                        'trades/{pair}',\n                    ],\n                },\n                // trading (private)\n                'private': {\n                    'post': [\n                        'getInfo',\n                        'TransHistory',\n                        'TradeHistory',\n                        'OrderHistory',\n                        'ActiveOrders',\n                        'Trade',\n                        'CancelOrder',\n                    ],\n                },\n                // deposit / withdraw (private)\n                'dwapi': {\n                    'post': [\n                        'getCryptoDepositAddress',\n                        'cryptoWithdraw',\n                        'fiatWithdraw',\n                        'getTransactionStatus',\n                        'getTransactions',\n                    ],\n                },\n            },\n        });\n    }\n\n    getBaseQuoteFromMarketId (id) {\n        let uppercase = id.toUpperCase ();\n        let base = uppercase.slice (0, 3);\n        let quote = uppercase.slice (3, 6);\n        base = this.commonCurrencyCode (base);\n        quote = this.commonCurrencyCode (quote);\n        return [ base, quote ];\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetInfo ();\n        let balances = response['return'];\n        let result = { 'info': balances };\n        let funds = balances['funds'];\n        let currencies = Object.keys (funds);\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let uppercase = currency.toUpperCase ();\n            uppercase = this.commonCurrencyCode (uppercase);\n            let account = {\n                'free': funds[currency],\n                'used': 0.0,\n                'total': balances['total'][currency],\n            };\n            account['used'] = account['total'] - account['free'];\n            result[uppercase] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['updated'] * 1000;\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'buy'),\n            'ask': this.safeFloat (ticker, 'sell'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': 1 / this.safeFloat (ticker, 'avg'),\n            'baseVolume': this.safeFloat (ticker, 'vol'),\n            'quoteVolume': this.safeFloat (ticker, 'vol_cur'),\n            'info': ticker,\n        };\n    }\n\n    getOrderIdKey () {\n        return 'orderId';\n    }\n\n    signBodyWithSecret (body) {\n        return this.decode (this.hmac (this.encode (body), this.encode (this.secret), 'sha512', 'base64'));\n    }\n\n    getVersionString () {\n        return ''; // they don't prepend version number to public URLs as other BTC-e clones do\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class exmo extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'exmo',\n            'name': 'EXMO',\n            'countries': [ 'ES', 'RU' ], // Spain, Russia\n            'rateLimit': 1000, // once every 350 ms ≈ 180 requests per minute ≈ 3 requests per second\n            'version': 'v1',\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766491-1b0ea956-5eda-11e7-9225-40d67b481b8d.jpg',\n                'api': 'https://api.exmo.com',\n                'www': 'https://exmo.me',\n                'doc': [\n                    'https://exmo.me/en/api_doc',\n                    'https://github.com/exmo-dev/exmo_api_lib/tree/master/nodejs',\n                ],\n                'fees': 'https://exmo.com/en/docs/fees',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'currency',\n                        'order_book',\n                        'pair_settings',\n                        'ticker',\n                        'trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'user_info',\n                        'order_create',\n                        'order_cancel',\n                        'user_open_orders',\n                        'user_trades',\n                        'user_cancelled_orders',\n                        'order_trades',\n                        'required_amount',\n                        'deposit_address',\n                        'withdraw_crypt',\n                        'withdraw_get_txid',\n                        'excode_create',\n                        'excode_load',\n                        'wallet_history',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.2 / 100,\n                    'taker': 0.2 / 100,\n                },\n                'funding': {\n                    'witdhraw': {\n                        'BTC': 0.001,\n                        'LTC': 0.01,\n                        'DOGE': 1,\n                        'DASH': 0.01,\n                        'ETH': 0.01,\n                        'WAVES': 0.001,\n                        'ZEC': 0.001,\n                        'USDT': 25,\n                        'XMR': 0.05,\n                        'XRP': 0.02,\n                        'KICK': 350,\n                        'ETC': 0.01,\n                        'BCH': 0.001,\n                    },\n                    'deposit': {\n                        'USDT': 15,\n                        'KICK': 50,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetPairSettings ();\n        let keys = Object.keys (markets);\n        let result = [];\n        for (let p = 0; p < keys.length; p++) {\n            let id = keys[p];\n            let market = markets[id];\n            let symbol = id.replace ('_', '/');\n            let [ base, quote ] = symbol.split ('/');\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'limits': {\n                    'amount': {\n                        'min': market['min_quantity'],\n                        'max': market['max_quantity'],\n                    },\n                    'price': {\n                        'min': market['min_price'],\n                        'max': market['max_price'],\n                    },\n                    'cost': {\n                        'min': market['min_amount'],\n                        'max': market['max_amount'],\n                    },\n                },\n                'precision': {\n                    'amount': 8,\n                    'price': 8,\n                },\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostUserInfo ();\n        let result = { 'info': response };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            if (currency in response['balances'])\n                account['free'] = parseFloat (response['balances'][currency]);\n            if (currency in response['reserved'])\n                account['used'] = parseFloat (response['reserved'][currency]);\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetOrderBook (this.extend ({\n            'pair': market['id'],\n        }, params));\n        let result = response[market['id']];\n        let orderbook = this.parseOrderBook (result, undefined, 'bid', 'ask');\n        return this.extend (orderbook, {\n            'bids': this.sortBy (orderbook['bids'], 0, true),\n            'asks': this.sortBy (orderbook['asks'], 0),\n        });\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['updated'] * 1000;\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy_price']),\n            'ask': parseFloat (ticker['sell_price']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last_trade']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': parseFloat (ticker['avg']),\n            'baseVolume': parseFloat (ticker['vol']),\n            'quoteVolume': parseFloat (ticker['vol_curr']),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetTicker (params);\n        let result = {};\n        let ids = Object.keys (response);\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = response[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetTicker (params);\n        let market = this.market (symbol);\n        return this.parseTicker (response[market['id']], market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['date'] * 1000;\n        return {\n            'id': trade['trade_id'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': undefined,\n            'type': undefined,\n            'side': trade['type'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['quantity']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTrades (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTrades (response[market['id']], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let prefix = '';\n        if (type === 'market')\n            prefix = 'market_';\n        if (typeof price === 'undefined')\n            price = 0;\n        let order = {\n            'pair': this.marketId (symbol),\n            'quantity': amount,\n            'price': price,\n            'type': prefix + side,\n        };\n        let response = await this.privatePostOrderCreate (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['order_id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostOrderCancel ({ 'order_id': id });\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let result = await this.privatePostWithdrawCrypt (this.extend ({\n            'amount': amount,\n            'currency': currency,\n            'address': address,\n        }, params));\n        return {\n            'info': result,\n            'id': result['task_id'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + path;\n        if (api === 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({ 'nonce': nonce }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Key': this.apiKey,\n                'Sign': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('result' in response) {\n            if (response['result'])\n                return response;\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class flowbtc extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'flowbtc',\n            'name': 'flowBTC',\n            'countries': 'BR', // Brazil\n            'version': 'v1',\n            'rateLimit': 1000,\n            'has': {\n                'CORS': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28162465-cd815d4c-67cf-11e7-8e57-438bea0523a2.jpg',\n                'api': 'https://api.flowbtc.com:8400/ajax',\n                'www': 'https://trader.flowbtc.com',\n                'doc': 'http://www.flowbtc.com.br/api/',\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'uid': true,\n            },\n            'api': {\n                'public': {\n                    'post': [\n                        'GetTicker',\n                        'GetTrades',\n                        'GetTradesByDate',\n                        'GetOrderBook',\n                        'GetProductPairs',\n                        'GetProducts',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'CreateAccount',\n                        'GetUserInfo',\n                        'SetUserInfo',\n                        'GetAccountInfo',\n                        'GetAccountTrades',\n                        'GetDepositAddresses',\n                        'Withdraw',\n                        'CreateOrder',\n                        'ModifyOrder',\n                        'CancelOrder',\n                        'CancelAllOrders',\n                        'GetAccountOpenOrders',\n                        'GetOrderFee',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicPostGetProductPairs ();\n        let markets = response['productPairs'];\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['name'];\n            let base = market['product1Label'];\n            let quote = market['product2Label'];\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetAccountInfo ();\n        let balances = response['currencies'];\n        let result = { 'info': response };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['name'];\n            let account = {\n                'free': balance['balance'],\n                'used': balance['hold'],\n                'total': 0.0,\n            };\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orderbook = await this.publicPostGetOrderBook (this.extend ({\n            'productPair': market['id'],\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'px', 'qty');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicPostGetTicker (this.extend ({\n            'productPair': market['id'],\n        }, params));\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume24hr']),\n            'quoteVolume': parseFloat (ticker['volume24hrProduct2']),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['unixtime'] * 1000;\n        let side = (trade['incomingOrderSide'] == 0) ? 'buy' : 'sell';\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': trade['tid'].toString (),\n            'order': undefined,\n            'type': undefined,\n            'side': side,\n            'price': trade['px'],\n            'amount': trade['qty'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicPostGetTrades (this.extend ({\n            'ins': market['id'],\n            'startIndex': -1,\n        }, params));\n        return this.parseTrades (response['trades'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let orderType = (type == 'market') ? 1 : 0;\n        let order = {\n            'ins': this.marketId (symbol),\n            'side': side,\n            'orderType': orderType,\n            'qty': amount,\n            'px': price,\n        };\n        let response = await this.privatePostCreateOrder (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['serverOrderId'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        if ('ins' in params) {\n            return await this.privatePostCancelOrder (this.extend ({\n                'serverOrderId': id,\n            }, params));\n        }\n        throw new ExchangeError (this.id + ' requires `ins` symbol parameter for cancelling an order');\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + path;\n        if (api == 'public') {\n            if (Object.keys (params).length) {\n                body = this.json (params);\n            }\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let auth = nonce.toString () + this.uid + this.apiKey;\n            let signature = this.hmac (this.encode (auth), this.encode (this.secret));\n            body = this.json (this.extend ({\n                'apiKey': this.apiKey,\n                'apiNonce': nonce,\n                'apiSig': signature.toUpperCase (),\n            }, params));\n            headers = {\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('isAccepted' in response)\n            if (response['isAccepted'])\n                return response;\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class foxbit extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'foxbit',\n            'name': 'FoxBit',\n            'countries': 'BR',\n            'has': {\n                'CORS': false,\n            },\n            'rateLimit': 1000,\n            'version': 'v1',\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27991413-11b40d42-647f-11e7-91ee-78ced874dd09.jpg',\n                'api': {\n                    'public': 'https://api.blinktrade.com/api',\n                    'private': 'https://api.blinktrade.com/tapi',\n                },\n                'www': 'https://foxbit.exchange',\n                'doc': 'https://blinktrade.com/docs',\n            },\n            'comment': 'Blinktrade API',\n            'api': {\n                'public': {\n                    'get': [\n                        '{currency}/ticker',    // ?crypto_currency=BTC\n                        '{currency}/orderbook', // ?crypto_currency=BTC\n                        '{currency}/trades',    // ?crypto_currency=BTC&since=<TIMESTAMP>&limit=<NUMBER>\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'D',   // order\n                        'F',   // cancel order\n                        'U2',  // balance\n                        'U4',  // my orders\n                        'U6',  // withdraw\n                        'U18', // deposit\n                        'U24', // confirm withdrawal\n                        'U26', // list withdrawals\n                        'U30', // list deposits\n                        'U34', // ledger\n                        'U70', // cancel withdrawal\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/VEF': { 'id': 'BTCVEF', 'symbol': 'BTC/VEF', 'base': 'BTC', 'quote': 'VEF', 'brokerId': 1, 'broker': 'SurBitcoin' },\n                'BTC/VND': { 'id': 'BTCVND', 'symbol': 'BTC/VND', 'base': 'BTC', 'quote': 'VND', 'brokerId': 3, 'broker': 'VBTC' },\n                'BTC/BRL': { 'id': 'BTCBRL', 'symbol': 'BTC/BRL', 'base': 'BTC', 'quote': 'BRL', 'brokerId': 4, 'broker': 'FoxBit' },\n                'BTC/PKR': { 'id': 'BTCPKR', 'symbol': 'BTC/PKR', 'base': 'BTC', 'quote': 'PKR', 'brokerId': 8, 'broker': 'UrduBit' },\n                'BTC/CLP': { 'id': 'BTCCLP', 'symbol': 'BTC/CLP', 'base': 'BTC', 'quote': 'CLP', 'brokerId': 9, 'broker': 'ChileBit' },\n            },\n        });\n    }\n\n    fetchBalance (params = {}) {\n        // todo parse balance\n        return this.privatePostU2 ({\n            'BalanceReqID': this.nonce (),\n        });\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let market = this.market (symbol);\n        let orderbook = await this.publicGetCurrencyOrderbook (this.extend ({\n            'currency': market['quote'],\n            'crypto_currency': market['base'],\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let market = this.market (symbol);\n        let ticker = await this.publicGetCurrencyTicker (this.extend ({\n            'currency': market['quote'],\n            'crypto_currency': market['base'],\n        }, params));\n        let timestamp = this.milliseconds ();\n        let lowercaseQuote = market['quote'].toLowerCase ();\n        let quoteVolume = 'vol_' + lowercaseQuote;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy']),\n            'ask': parseFloat (ticker['sell']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['vol']),\n            'quoteVolume': parseFloat (ticker[quoteVolume]),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['date'] * 1000;\n        return {\n            'id': trade['tid'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['side'],\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetCurrencyTrades (this.extend ({\n            'currency': market['quote'],\n            'crypto_currency': market['base'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type == 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let market = this.market (symbol);\n        let orderSide = (side == 'buy') ? '1' : '2';\n        let order = {\n            'ClOrdID': this.nonce (),\n            'Symbol': market['id'],\n            'Side': orderSide,\n            'OrdType': '2',\n            'Price': price,\n            'OrderQty': amount,\n            'BrokerID': market['brokerId'],\n        };\n        let response = await this.privatePostD (this.extend (order, params));\n        let indexed = this.indexBy (response['Responses'], 'MsgType');\n        let execution = indexed['8'];\n        return {\n            'info': response,\n            'id': execution['OrderID'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostF (this.extend ({\n            'ClOrdID': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api] + '/' + this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let request = this.extend ({ 'MsgType': path }, query);\n            body = this.json (request);\n            headers = {\n                'APIKey': this.apiKey,\n                'Nonce': nonce,\n                'Signature': this.hmac (this.encode (nonce), this.encode (this.secret)),\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('Status' in response)\n            if (response['Status'] != 200)\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class fybse extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'fybse',\n            'name': 'FYB-SE',\n            'countries': 'SE', // Sweden\n            'has': {\n                'CORS': false,\n            },\n            'rateLimit': 1500,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766512-31019772-5edb-11e7-8241-2e675e6797f1.jpg',\n                'api': 'https://www.fybse.se/api/SEK',\n                'www': 'https://www.fybse.se',\n                'doc': 'http://docs.fyb.apiary.io',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'ticker',\n                        'tickerdetailed',\n                        'orderbook',\n                        'trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'test',\n                        'getaccinfo',\n                        'getpendingorders',\n                        'getorderhistory',\n                        'cancelpendingorder',\n                        'placeorder',\n                        'withdraw',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/SEK': { 'id': 'SEK', 'symbol': 'BTC/SEK', 'base': 'BTC', 'quote': 'SEK' },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let balance = await this.privatePostGetaccinfo ();\n        let btc = parseFloat (balance['btcBal']);\n        let symbol = this.symbols[0];\n        let quote = this.markets[symbol]['quote'];\n        let lowercase = quote.toLowerCase () + 'Bal';\n        let fiat = parseFloat (balance[lowercase]);\n        let crypto = {\n            'free': btc,\n            'used': 0.0,\n            'total': btc,\n        };\n        let result = { 'BTC': crypto };\n        result[quote] = {\n            'free': fiat,\n            'used': 0.0,\n            'total': fiat,\n        };\n        result['info'] = balance;\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetOrderbook (params);\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetTickerdetailed (params);\n        let timestamp = this.milliseconds ();\n        let last = undefined;\n        let volume = undefined;\n        if ('last' in ticker)\n            last = parseFloat (ticker['last']);\n        if ('vol' in ticker)\n            volume = parseFloat (ticker['vol']);\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': last,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': volume,\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        return {\n            'info': trade,\n            'id': trade['tid'].toString (),\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetTrades (params);\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let response = await this.privatePostPlaceorder (this.extend ({\n            'qty': amount,\n            'price': price,\n            'type': side[0].toUpperCase (),\n        }, params));\n        return {\n            'info': response,\n            'id': response['pending_oid'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelpendingorder ({ 'orderNo': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + path;\n        if (api == 'public') {\n            url += '.json';\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({ 'timestamp': nonce }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'key': this.apiKey,\n                'sig': this.hmac (this.encode (body), this.encode (this.secret), 'sha1'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (api == 'private')\n            if ('error' in response)\n                if (response['error'])\n                    throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst fybse = require ('./fybse.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class fybsg extends fybse {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'fybsg',\n            'name': 'FYB-SG',\n            'countries': 'SG', // Singapore\n            'has': {\n                'CORS': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766513-3364d56a-5edb-11e7-9e6b-d5898bb89c81.jpg',\n                'api': 'https://www.fybsg.com/api/SGD',\n                'www': 'https://www.fybsg.com',\n                'doc': 'http://docs.fyb.apiary.io',\n            },\n            'markets': {\n                'BTC/SGD': { 'id': 'SGD', 'symbol': 'BTC/SGD', 'base': 'BTC', 'quote': 'SGD' },\n            },\n        });\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, AuthenticationError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class gatecoin extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'gatecoin',\n            'name': 'Gatecoin',\n            'rateLimit': 2000,\n            'countries': 'HK', // Hong Kong\n            'comment': 'a regulated/licensed exchange',\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n                'fetchOHLCV': true,\n            },\n            'timeframes': {\n                '1m': '1m',\n                '15m': '15m',\n                '1h': '1h',\n                '6h': '6h',\n                '1d': '24h',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28646817-508457f2-726c-11e7-9eeb-3528d2413a58.jpg',\n                'api': 'https://api.gatecoin.com',\n                'www': 'https://gatecoin.com',\n                'doc': [\n                    'https://gatecoin.com/api',\n                    'https://github.com/Gatecoin/RESTful-API-Implementation',\n                    'https://api.gatecoin.com/swagger-ui/index.html',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'Public/ExchangeRate', // Get the exchange rates\n                        'Public/LiveTicker', // Get live ticker for all currency\n                        'Public/LiveTicker/{CurrencyPair}', // Get live ticker by currency\n                        'Public/LiveTickers', // Get live ticker for all currency\n                        'Public/MarketDepth/{CurrencyPair}', // Gets prices and market depth for the currency pair.\n                        'Public/NetworkStatistics/{DigiCurrency}', // Get the network status of a specific digital currency\n                        'Public/StatisticHistory/{DigiCurrency}/{Typeofdata}', // Get the historical data of a specific digital currency\n                        'Public/TickerHistory/{CurrencyPair}/{Timeframe}', // Get ticker history\n                        'Public/Transactions/{CurrencyPair}', // Gets recent transactions\n                        'Public/TransactionsHistory/{CurrencyPair}', // Gets all transactions\n                        'Reference/BusinessNatureList', // Get the business nature list.\n                        'Reference/Countries', // Get the country list.\n                        'Reference/Currencies', // Get the currency list.\n                        'Reference/CurrencyPairs', // Get the currency pair list.\n                        'Reference/CurrentStatusList', // Get the current status list.\n                        'Reference/IdentydocumentTypes', // Get the different types of identity documents possible.\n                        'Reference/IncomeRangeList', // Get the income range list.\n                        'Reference/IncomeSourceList', // Get the income source list.\n                        'Reference/VerificationLevelList', // Get the verif level list.\n                        'Stream/PublicChannel', // Get the public pubnub channel list\n                    ],\n                    'post': [\n                        'Export/Transactions', // Request a export of all trades from based on currencypair, start date and end date\n                        'Ping', // Post a string, then get it back.\n                        'Public/Unsubscribe/{EmailCode}', // Lets the user unsubscribe from emails\n                        'RegisterUser', // Initial trader registration.\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'Account/CorporateData', // Get corporate account data\n                        'Account/DocumentAddress', // Check if residence proof uploaded\n                        'Account/DocumentCorporation', // Check if registered document uploaded\n                        'Account/DocumentID', // Check if ID document copy uploaded\n                        'Account/DocumentInformation', // Get Step3 Data\n                        'Account/Email', // Get user email\n                        'Account/FeeRate', // Get fee rate of logged in user\n                        'Account/Level', // Get verif level of logged in user\n                        'Account/PersonalInformation', // Get Step1 Data\n                        'Account/Phone', // Get user phone number\n                        'Account/Profile', // Get trader profile\n                        'Account/Questionnaire', // Fill the questionnaire\n                        'Account/Referral', // Get referral information\n                        'Account/ReferralCode', // Get the referral code of the logged in user\n                        'Account/ReferralNames', // Get names of referred traders\n                        'Account/ReferralReward', // Get referral reward information\n                        'Account/ReferredCode', // Get referral code\n                        'Account/ResidentInformation', // Get Step2 Data\n                        'Account/SecuritySettings', // Get verif details of logged in user\n                        'Account/User', // Get all user info\n                        'APIKey/APIKey', // Get API Key for logged in user\n                        'Auth/ConnectionHistory', // Gets connection history of logged in user\n                        'Balance/Balances', // Gets the available balance for each currency for the logged in account.\n                        'Balance/Balances/{Currency}', // Gets the available balance for s currency for the logged in account.\n                        'Balance/Deposits', // Get all account deposits, including wire and digital currency, of the logged in user\n                        'Balance/Withdrawals', // Get all account withdrawals, including wire and digital currency, of the logged in user\n                        'Bank/Accounts/{Currency}/{Location}', // Get internal bank account for deposit\n                        'Bank/Transactions', // Get all account transactions of the logged in user\n                        'Bank/UserAccounts', // Gets all the bank accounts related to the logged in user.\n                        'Bank/UserAccounts/{Currency}', // Gets all the bank accounts related to the logged in user.\n                        'ElectronicWallet/DepositWallets', // Gets all crypto currency addresses related deposits to the logged in user.\n                        'ElectronicWallet/DepositWallets/{DigiCurrency}', // Gets all crypto currency addresses related deposits to the logged in user by currency.\n                        'ElectronicWallet/Transactions', // Get all digital currency transactions of the logged in user\n                        'ElectronicWallet/Transactions/{DigiCurrency}', // Get all digital currency transactions of the logged in user\n                        'ElectronicWallet/UserWallets', // Gets all external digital currency addresses related to the logged in user.\n                        'ElectronicWallet/UserWallets/{DigiCurrency}', // Gets all external digital currency addresses related to the logged in user by currency.\n                        'Info/ReferenceCurrency', // Get user's reference currency\n                        'Info/ReferenceLanguage', // Get user's reference language\n                        'Notification/Messages', // Get from oldest unread + 3 read message to newest messages\n                        'Trade/Orders', // Gets open orders for the logged in trader.\n                        'Trade/Orders/{OrderID}', // Gets an order for the logged in trader.\n                        'Trade/StopOrders', // Gets all stop orders for the logged in trader. Max 1000 record.\n                        'Trade/StopOrdersHistory', // Gets all stop orders for the logged in trader. Max 1000 record.\n                        'Trade/Trades', // Gets all transactions of logged in user\n                        'Trade/UserTrades', // Gets all transactions of logged in user\n                    ],\n                    'post': [\n                        'Account/DocumentAddress', // Upload address proof document\n                        'Account/DocumentCorporation', // Upload registered document document\n                        'Account/DocumentID', // Upload ID document copy\n                        'Account/Email/RequestVerify', // Request for verification email\n                        'Account/Email/Verify', // Verification email\n                        'Account/GoogleAuth', // Enable google auth\n                        'Account/Level', // Request verif level of logged in user\n                        'Account/Questionnaire', // Fill the questionnaire\n                        'Account/Referral', // Post a referral email\n                        'APIKey/APIKey', // Create a new API key for logged in user\n                        'Auth/ChangePassword', // Change password.\n                        'Auth/ForgotPassword', // Request reset password\n                        'Auth/ForgotUserID', // Request user id\n                        'Auth/Login', // Trader session log in.\n                        'Auth/Logout', // Logout from the current session.\n                        'Auth/LogoutOtherSessions', // Logout other sessions.\n                        'Auth/ResetPassword', // Reset password\n                        'Bank/Transactions', // Request a transfer from the traders account of the logged in user. This is only available for bank account\n                        'Bank/UserAccounts', // Add an account the logged in user\n                        'ElectronicWallet/DepositWallets/{DigiCurrency}', // Add an digital currency addresses to the logged in user.\n                        'ElectronicWallet/Transactions/Deposits/{DigiCurrency}', // Get all internal digital currency transactions of the logged in user\n                        'ElectronicWallet/Transactions/Withdrawals/{DigiCurrency}', // Get all external digital currency transactions of the logged in user\n                        'ElectronicWallet/UserWallets/{DigiCurrency}', // Add an external digital currency addresses to the logged in user.\n                        'ElectronicWallet/Withdrawals/{DigiCurrency}', // Request a transfer from the traders account to an external address. This is only available for crypto currencies.\n                        'Notification/Messages', // Mark all as read\n                        'Notification/Messages/{ID}', // Mark as read\n                        'Trade/Orders', // Place an order at the exchange.\n                        'Trade/StopOrders', // Place a stop order at the exchange.\n                    ],\n                    'put': [\n                        'Account/CorporateData', // Update user company data for corporate account\n                        'Account/DocumentID', // Update ID document meta data\n                        'Account/DocumentInformation', // Update Step3 Data\n                        'Account/Email', // Update user email\n                        'Account/PersonalInformation', // Update Step1 Data\n                        'Account/Phone', // Update user phone number\n                        'Account/Questionnaire', // update the questionnaire\n                        'Account/ReferredCode', // Update referral code\n                        'Account/ResidentInformation', // Update Step2 Data\n                        'Account/SecuritySettings', // Update verif details of logged in user\n                        'Account/User', // Update all user info\n                        'Bank/UserAccounts', // Update the label of existing user bank accounnt\n                        'ElectronicWallet/DepositWallets/{DigiCurrency}/{AddressName}', // Update the name of an address\n                        'ElectronicWallet/UserWallets/{DigiCurrency}', // Update the name of an external address\n                        'Info/ReferenceCurrency', // User's reference currency\n                        'Info/ReferenceLanguage', // Update user's reference language\n                    ],\n                    'delete': [\n                        'APIKey/APIKey/{PublicKey}', // Remove an API key\n                        'Bank/Transactions/{RequestID}', // Delete pending account withdraw of the logged in user\n                        'Bank/UserAccounts/{Currency}/{Label}', // Delete an account of the logged in user\n                        'ElectronicWallet/DepositWallets/{DigiCurrency}/{AddressName}', // Delete an digital currency addresses related to the logged in user.\n                        'ElectronicWallet/UserWallets/{DigiCurrency}/{AddressName}', // Delete an external digital currency addresses related to the logged in user.\n                        'Trade/Orders', // Cancels all existing order\n                        'Trade/Orders/{OrderID}', // Cancels an existing order\n                        'Trade/StopOrders', // Cancels all existing stop orders\n                        'Trade/StopOrders/{ID}', // Cancels an existing stop order\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.0025,\n                    'taker': 0.0035,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicGetPublicLiveTickers ();\n        let markets = response['tickers'];\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['currencyPair'];\n            let base = id.slice (0, 3);\n            let quote = id.slice (3, 6);\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetBalanceBalances ();\n        let balances = response['balances'];\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let account = {\n                'free': balance['availableBalance'],\n                'used': this.sum (\n                    balance['pendingIncoming'],\n                    balance['pendingOutgoing'],\n                    balance['openOrder']),\n                'total': balance['balance'],\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orderbook = await this.publicGetPublicMarketDepthCurrencyPair (this.extend ({\n            'CurrencyPair': market['id'],\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'volume');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = parseInt (ticker['createDateTime']) * 1000;\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let baseVolume = parseFloat (ticker['volume']);\n        let vwap = parseFloat (ticker['vwap']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': vwap,\n            'open': parseFloat (ticker['open']),\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetPublicLiveTickers (params);\n        let tickers = response['tickers'];\n        let result = {};\n        for (let t = 0; t < tickers.length; t++) {\n            let ticker = tickers[t];\n            let id = ticker['currencyPair'];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetPublicLiveTickerCurrencyPair (this.extend ({\n            'CurrencyPair': market['id'],\n        }, params));\n        let ticker = response['ticker'];\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let side = undefined;\n        let order = undefined;\n        if ('way' in trade) {\n            side = (trade['way'] == 'bid') ? 'buy' : 'sell';\n            let orderId = trade['way'] + 'OrderId';\n            order = trade[orderId];\n        }\n        let timestamp = parseInt (trade['transactionTime']) * 1000;\n        if (!market)\n            market = this.markets_by_id[trade['currencyPair']];\n        return {\n            'info': trade,\n            'id': trade['transactionId'].toString (),\n            'order': order,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': side,\n            'price': trade['price'],\n            'amount': trade['quantity'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetPublicTransactionsCurrencyPair (this.extend ({\n            'CurrencyPair': market['id'],\n        }, params));\n        return this.parseTrades (response['transactions'], market, since, limit);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            parseInt (ohlcv['createDateTime']) * 1000,\n            ohlcv['open'],\n            ohlcv['high'],\n            ohlcv['low'],\n            undefined,\n            ohlcv['volume'],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'CurrencyPair': market['id'],\n            'Timeframe': this.timeframes[timeframe],\n        };\n        if (limit)\n            request['Count'] = limit;\n        request = this.extend (request, params);\n        let response = await this.publicGetPublicTickerHistoryCurrencyPairTimeframe (request);\n        return this.parseOHLCVs (response['tickers'], market, timeframe, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'Code': this.marketId (symbol),\n            'Way': (side == 'buy') ? 'Bid' : 'Ask',\n            'Amount': amount,\n        };\n        if (type == 'limit')\n            order['Price'] = price;\n        if (this.twofa) {\n            if ('ValidationCode' in params)\n                order['ValidationCode'] = params['ValidationCode'];\n            else\n                throw new AuthenticationError (this.id + ' two-factor authentication requires a missing ValidationCode parameter');\n        }\n        let response = await this.privatePostTradeOrders (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['clOrderId'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privateDeleteTradeOrdersOrderID ({ 'OrderID': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let nonceString = nonce.toString ();\n            let contentType = (method == 'GET') ? '' : 'application/json';\n            let auth = method + url + contentType + nonceString;\n            auth = auth.toLowerCase ();\n            let signature = this.hmac (this.encode (auth), this.encode (this.secret), 'sha256', 'base64');\n            headers = {\n                'API_PUBLIC_KEY': this.apiKey,\n                'API_REQUEST_SIGNATURE': this.decode (signature),\n                'API_REQUEST_DATE': nonceString,\n            };\n            if (method != 'GET') {\n                headers['Content-Type'] = contentType;\n                body = this.json (this.extend ({ 'nonce': nonce }, params));\n            }\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('responseStatus' in response)\n            if ('message' in response['responseStatus'])\n                if (response['responseStatus']['message'] == 'OK')\n                    return response;\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst bter = require ('./bter.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class gateio extends bter {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'gateio',\n            'name': 'Gate.io',\n            'countries': 'CN',\n            'rateLimit': 1000,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/31784029-0313c702-b509-11e7-9ccc-bc0da6a0e435.jpg',\n                'api': {\n                    'public': 'https://data.gate.io/api',\n                    'private': 'https://data.gate.io/api',\n                },\n                'www': 'https://gate.io/',\n                'doc': 'https://gate.io/api2',\n                'fees': 'https://gate.io/fee',\n            },\n        });\n    }\n\n    parseTrade (trade, market) {\n        // exchange reports local time (UTC+8)\n        let timestamp = this.parse8601 (trade['date']) - 8 * 60 * 60 * 1000;\n        return {\n            'id': trade['tradeID'],\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['type'],\n            'price': trade['rate'],\n            'amount': this.safeFloat (trade, 'amount'),\n        };\n    }\n}\n","'use strict';\n\n// ----------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InvalidOrder, AuthenticationError, NotSupported } = require ('./base/errors');\n\n// ----------------------------------------------------------------------------\n\nmodule.exports = class gdax extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'gdax',\n            'name': 'GDAX',\n            'countries': 'US',\n            'rateLimit': 1000,\n            'userAgent': this.userAgents['chrome'],\n            'has': {\n                'CORS': true,\n                'fetchOHLCV': true,\n                'deposit': true,\n                'withdraw': true,\n                'fetchOrder': true,\n                'fetchOrders': true,\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': true,\n            },\n            'timeframes': {\n                '1m': 60,\n                '5m': 300,\n                '15m': 900,\n                '30m': 1800,\n                '1h': 3600,\n                '2h': 7200,\n                '4h': 14400,\n                '12h': 43200,\n                '1d': 86400,\n                '1w': 604800,\n                '1M': 2592000,\n                '1y': 31536000,\n            },\n            'urls': {\n                'test': 'https://api-public.sandbox.gdax.com',\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766527-b1be41c6-5edb-11e7-95f6-5b496c469e2c.jpg',\n                'api': 'https://api.gdax.com',\n                'www': 'https://www.gdax.com',\n                'doc': 'https://docs.gdax.com',\n                'fees': [\n                    'https://www.gdax.com/fees',\n                    'https://support.gdax.com/customer/en/portal/topics/939402-depositing-and-withdrawing-funds/articles',\n                ],\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'password': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'currencies',\n                        'products',\n                        'products/{id}/book',\n                        'products/{id}/candles',\n                        'products/{id}/stats',\n                        'products/{id}/ticker',\n                        'products/{id}/trades',\n                        'time',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'accounts',\n                        'accounts/{id}',\n                        'accounts/{id}/holds',\n                        'accounts/{id}/ledger',\n                        'coinbase-accounts',\n                        'fills',\n                        'funding',\n                        'orders',\n                        'orders/{id}',\n                        'payment-methods',\n                        'position',\n                        'reports/{id}',\n                        'users/self/trailing-volume',\n                    ],\n                    'post': [\n                        'deposits/coinbase-account',\n                        'deposits/payment-method',\n                        'funding/repay',\n                        'orders',\n                        'position/close',\n                        'profiles/margin-transfer',\n                        'reports',\n                        'withdrawals/coinbase',\n                        'withdrawals/crypto',\n                        'withdrawals/payment-method',\n                    ],\n                    'delete': [\n                        'orders',\n                        'orders/{id}',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': true, // complicated tier system per coin\n                    'percentage': true,\n                    'maker': 0.0,\n                    'taker': 0.30 / 100, // worst-case scenario: https://www.gdax.com/fees/BTC-USD\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BCH': 0,\n                        'BTC': 0,\n                        'LTC': 0,\n                        'ETH': 0,\n                        'EUR': 0.15,\n                        'USD': 25,\n                    },\n                    'deposit': {\n                        'BCH': 0,\n                        'BTC': 0,\n                        'LTC': 0,\n                        'ETH': 0,\n                        'EUR': 0.15,\n                        'USD': 10,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetProducts ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['id'];\n            let base = market['base_currency'];\n            let quote = market['quote_currency'];\n            let symbol = base + '/' + quote;\n            let amountLimits = {\n                'min': market['base_min_size'],\n                'max': market['base_max_size'],\n            };\n            let priceLimits = {\n                'min': market['quote_increment'],\n                'max': undefined,\n            };\n            let costLimits = {\n                'min': priceLimits['min'],\n                'max': undefined,\n            };\n            let limits = {\n                'amount': amountLimits,\n                'price': priceLimits,\n                'cost': costLimits,\n            };\n            let precision = {\n                'amount': -Math.log10 (parseFloat (amountLimits['min'])),\n                'price': -Math.log10 (parseFloat (priceLimits['min'])),\n            };\n            let taker = this.fees['trading']['taker'];\n            if ((base === 'ETH') || (base === 'LTC')) {\n                taker = 0.003;\n            }\n            let active = market['status'] === 'online';\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n                'precision': precision,\n                'limits': limits,\n                'taker': taker,\n                'active': active,\n            }));\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privateGetAccounts ();\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let account = {\n                'free': parseFloat (balance['available']),\n                'used': parseFloat (balance['hold']),\n                'total': parseFloat (balance['balance']),\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetProductsIdBook (this.extend ({\n            'id': this.marketId (symbol),\n            'level': 2, // 1 best bidask, 2 aggregated, 3 full\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = this.extend ({\n            'id': market['id'],\n        }, params);\n        let ticker = await this.publicGetProductsIdTicker (request);\n        let timestamp = this.parse8601 (ticker['time']);\n        let bid = undefined;\n        let ask = undefined;\n        if ('bid' in ticker)\n            bid = parseFloat (ticker['bid']);\n        if ('ask' in ticker)\n            ask = parseFloat (ticker['ask']);\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': bid,\n            'ask': ask,\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'price'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = this.parse8601 (trade['time']);\n        let side = (trade['side'] === 'buy') ? 'sell' : 'buy';\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let fee = undefined;\n        if ('fill_fees' in trade) {\n            fee = {\n                'cost': parseFloat (trade['fill_fees']),\n                'currency': market['quote'],\n            };\n        }\n        return {\n            'id': trade['trade_id'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': undefined,\n            'side': side,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['size']),\n            'fee': fee,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetProductsIdTrades (this.extend ({\n            'id': market['id'], // fixes issue #2\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv[0] * 1000,\n            ohlcv[3],\n            ohlcv[2],\n            ohlcv[1],\n            ohlcv[4],\n            ohlcv[5],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let granularity = this.timeframes[timeframe];\n        let request = {\n            'id': market['id'],\n            'granularity': granularity,\n        };\n        if (since) {\n            request['start'] = this.YmdHMS (since);\n            if (!limit)\n                limit = 200; // max = 200\n            request['end'] = this.YmdHMS (this.sum (limit * granularity * 1000, since));\n        }\n        let response = await this.publicGetProductsIdCandles (this.extend (request, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    async fetchTime () {\n        let response = this.publicGetTime ();\n        return this.parse8601 (response['iso']);\n    }\n\n    parseOrderStatus (status) {\n        let statuses = {\n            'pending': 'open',\n            'active': 'open',\n            'open': 'open',\n            'done': 'closed',\n            'canceled': 'canceled',\n        };\n        return this.safeString (statuses, status, status);\n    }\n\n    parseOrder (order, market = undefined) {\n        let timestamp = this.parse8601 (order['created_at']);\n        let symbol = undefined;\n        if (!market) {\n            if (order['product_id'] in this.markets_by_id)\n                market = this.markets_by_id[order['product_id']];\n        }\n        let status = this.parseOrderStatus (order['status']);\n        let price = this.safeFloat (order, 'price');\n        let amount = this.safeFloat (order, 'size');\n        let filled = this.safeFloat (order, 'filled_size');\n        let remaining = amount - filled;\n        let cost = this.safeFloat (order, 'executed_value');\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'id': order['id'],\n            'info': order,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': status,\n            'symbol': symbol,\n            'type': order['type'],\n            'side': order['side'],\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'fee': undefined,\n        };\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetOrdersId (this.extend ({\n            'id': id,\n        }, params));\n        return this.parseOrder (response);\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            'status': 'all',\n        };\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['product_id'] = market['id'];\n        }\n        let response = await this.privateGetOrders (this.extend (request, params));\n        return this.parseOrders (response, market, since, limit);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['product_id'] = market['id'];\n        }\n        let response = await this.privateGetOrders (this.extend (request, params));\n        return this.parseOrders (response, market, since, limit);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            'status': 'done',\n        };\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['product_id'] = market['id'];\n        }\n        let response = await this.privateGetOrders (this.extend (request, params));\n        return this.parseOrders (response, market, since, limit);\n    }\n\n    async createOrder (market, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        // let oid = this.nonce ().toString ();\n        let order = {\n            'product_id': this.marketId (market),\n            'side': side,\n            'size': amount,\n            'type': type,\n        };\n        if (type === 'limit')\n            order['price'] = price;\n        let response = await this.privatePostOrders (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privateDeleteOrdersId ({ 'id': id });\n    }\n\n    async getPaymentMethods () {\n        let response = await this.privateGetPaymentMethods ();\n        return response;\n    }\n\n    async deposit (currency, amount, address, params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            'currency': currency,\n            'amount': amount,\n        };\n        let method = 'privatePostDeposits';\n        if ('payment_method_id' in params) {\n            // deposit from a payment_method, like a bank account\n            method += 'PaymentMethod';\n        } else if ('coinbase_account_id' in params) {\n            // deposit into GDAX account from a Coinbase account\n            method += 'CoinbaseAccount';\n        } else {\n            // deposit methodotherwise we did not receive a supported deposit location\n            // relevant docs link for the Googlers\n            // https://docs.gdax.com/#deposits\n            throw new NotSupported (this.id + ' deposit() requires one of `coinbase_account_id` or `payment_method_id` extra params');\n        }\n        let response = await this[method] (this.extend (request, params));\n        if (!response)\n            throw new ExchangeError (this.id + ' deposit() error: ' + this.json (response));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            'currency': currency,\n            'amount': amount,\n        };\n        let method = 'privatePostWithdrawals';\n        if ('payment_method_id' in params) {\n            method += 'PaymentMethod';\n        } else if ('coinbase_account_id' in params) {\n            method += 'CoinbaseAccount';\n        } else {\n            method += 'Crypto';\n            request['crypto_address'] = address;\n        }\n        let response = await this[method] (this.extend (request, params));\n        if (!response)\n            throw new ExchangeError (this.id + ' withdraw() error: ' + this.json (response));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let request = '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (method === 'GET') {\n            if (Object.keys (query).length)\n                request += '?' + this.urlencode (query);\n        }\n        let url = this.urls['api'] + request;\n        if (api === 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let payload = '';\n            if (method !== 'GET') {\n                if (Object.keys (query).length) {\n                    body = this.json (query);\n                    payload = body;\n                }\n            }\n            // let payload = (body) ? body : '';\n            let what = nonce + method + request + payload;\n            let secret = this.base64ToBinary (this.secret);\n            let signature = this.hmac (this.encode (what), secret, 'sha256', 'base64');\n            headers = {\n                'CB-ACCESS-KEY': this.apiKey,\n                'CB-ACCESS-SIGN': this.decode (signature),\n                'CB-ACCESS-TIMESTAMP': nonce,\n                'CB-ACCESS-PASSPHRASE': this.password,\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code === 400) {\n            if (body[0] === '{') {\n                let response = JSON.parse (body);\n                let message = response['message'];\n                if (message.indexOf ('price too small') >= 0) {\n                    throw new InvalidOrder (this.id + ' ' + message);\n                } else if (message.indexOf ('price too precise') >= 0) {\n                    throw new InvalidOrder (this.id + ' ' + message);\n                } else if (message === 'Invalid API Key') {\n                    throw new AuthenticationError (this.id + ' ' + message);\n                }\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n            }\n            throw new ExchangeError (this.id + ' ' + body);\n        }\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('message' in response) {\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class gemini extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'gemini',\n            'name': 'Gemini',\n            'countries': 'US',\n            'rateLimit': 1500, // 200 for private API\n            'version': 'v1',\n            'has': {\n                'CORS': false,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27816857-ce7be644-6096-11e7-82d6-3c257263229c.jpg',\n                'api': 'https://api.gemini.com',\n                'www': 'https://gemini.com',\n                'doc': [\n                    'https://docs.gemini.com/rest-api',\n                    'https://docs.sandbox.gemini.com',\n                ],\n                'test': 'https://api.sandbox.gemini.com',\n                'fees': [\n                    'https://gemini.com/fee-schedule/',\n                    'https://gemini.com/transfer-fees/',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'symbols',\n                        'pubticker/{symbol}',\n                        'book/{symbol}',\n                        'trades/{symbol}',\n                        'auction/{symbol}',\n                        'auction/{symbol}/history',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'order/new',\n                        'order/cancel',\n                        'order/cancel/session',\n                        'order/cancel/all',\n                        'order/status',\n                        'orders',\n                        'mytrades',\n                        'tradevolume',\n                        'balances',\n                        'deposit/{currency}/newAddress',\n                        'withdraw/{currency}',\n                        'heartbeat',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetSymbols ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let id = markets[p];\n            let market = id;\n            let uppercase = market.toUpperCase ();\n            let base = uppercase.slice (0, 3);\n            let quote = uppercase.slice (3, 6);\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n                'taker': 0.0025,\n            });\n        }\n        return result;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetBookSymbol (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'amount');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetPubtickerSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        let timestamp = ticker['volume']['timestamp'];\n        let baseVolume = market['base'];\n        let quoteVolume = market['quote'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume'][baseVolume]),\n            'quoteVolume': parseFloat (ticker['volume'][quoteVolume]),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['timestampms'];\n        return {\n            'id': trade['tid'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['type'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTradesSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privatePostBalances ();\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let account = {\n                'free': parseFloat (balance['available']),\n                'used': 0.0,\n                'total': parseFloat (balance['amount']),\n            };\n            account['used'] = account['total'] - account['free'];\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        if (type === 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let nonce = this.nonce ();\n        let order = {\n            'client_order_id': nonce.toString (),\n            'symbol': this.marketId (symbol),\n            'amount': amount.toString (),\n            'price': price.toString (),\n            'side': side,\n            'type': 'exchange limit', // gemini allows limit orders only\n        };\n        let response = await this.privatePostOrderNew (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['order_id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelOrder ({ 'order_id': id });\n    }\n\n    async withdraw (code, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let currency = this.currency (code);\n        let response = await this.privatePostWithdrawCurrency (this.extend ({\n            'currency': currency['id'],\n            'amount': amount,\n            'address': address,\n        }, params));\n        return {\n            'info': response,\n            'id': this.safeString (response, 'txHash'),\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = '/' + this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let request = this.extend ({\n                'request': url,\n                'nonce': nonce,\n            }, query);\n            let payload = this.json (request);\n            payload = this.stringToBase64 (this.encode (payload));\n            let signature = this.hmac (payload, this.encode (this.secret), 'sha384');\n            headers = {\n                'Content-Type': 'text/plain',\n                'X-GEMINI-APIKEY': this.apiKey,\n                'X-GEMINI-PAYLOAD': this.decode (payload),\n                'X-GEMINI-SIGNATURE': signature,\n            };\n        }\n        url = this.urls['api'] + url;\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('result' in response)\n            if (response['result'] === 'error')\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst _1btcxe = require ('./_1btcxe.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class getbtc extends _1btcxe {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'getbtc',\n            'name': 'GetBTC',\n            'countries': [ 'VC', 'RU' ], // Saint Vincent and the Grenadines, Russia, CIS\n            'rateLimit': 1000,\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/33801902-03c43462-dd7b-11e7-992e-077e4cd015b9.jpg',\n                'api': 'https://getbtc.org/api',\n                'www': 'https://getbtc.org',\n                'doc': 'https://getbtc.org/api-docs.php',\n            },\n            'markets': {\n                'BTC/EUR': { 'id': 'EUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR', 'precision': { 'amount': 8, 'price': 8 }, 'lot': 0.00000001, 'limits': { 'amount': { 'min': 0.00000001, 'max': undefined }, 'price': { 'min': 0.00000001, 'max': undefined }}},\n                'BTC/RUB': { 'id': 'RUB', 'symbol': 'BTC/RUB', 'base': 'BTC', 'quote': 'RUB', 'precision': { 'amount': 8, 'price': 8 }, 'lot': 0.00000001, 'limits': { 'amount': { 'min': 0.00000001, 'max': undefined }, 'price': { 'min': 0.00000001, 'max': undefined }}},\n                'BTC/USD': { 'id': 'USD', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD', 'precision': { 'amount': 8, 'price': 8 }, 'lot': 0.00000001, 'limits': { 'amount': { 'min': 0.00000001, 'max': undefined }, 'price': { 'min': 0.00000001, 'max': undefined }}},\n            },\n            'fees': {\n                'trading': {\n                    'taker': 0.20 / 100,\n                    'maker': 0.20 / 100,\n                },\n            },\n        });\n    }\n}\n","'use strict';\n\n// ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InsufficientFunds, OrderNotFound } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class hitbtc extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'hitbtc',\n            'name': 'HitBTC',\n            'countries': 'UK',\n            'rateLimit': 1500,\n            'version': '1',\n            'has': {\n                'CORS': false,\n                'fetchOrder': true,\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': true,\n                'withdraw': true\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766555-8eaec20e-5edc-11e7-9c5b-6dc69fc42f5e.jpg',\n                'api': 'http://api.hitbtc.com',\n                'www': 'https://hitbtc.com',\n                'doc': 'https://github.com/hitbtc-com/hitbtc-api/blob/master/APIv1.md',\n                'fees': [\n                    'https://hitbtc.com/fees-and-limits',\n                    'https://support.hitbtc.com/hc/en-us/articles/115005148605-Fees-and-limits',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        '{symbol}/orderbook',\n                        '{symbol}/ticker',\n                        '{symbol}/trades',\n                        '{symbol}/trades/recent',\n                        'symbols',\n                        'ticker',\n                        'time',\n                    ],\n                },\n                'trading': {\n                    'get': [\n                        'balance',\n                        'orders/active',\n                        'orders/recent',\n                        'order',\n                        'trades/by/order',\n                        'trades',\n                    ],\n                    'post': [\n                        'new_order',\n                        'cancel_order',\n                        'cancel_orders',\n                    ],\n                },\n                'payment': {\n                    'get': [\n                        'balance',\n                        'address/{currency}',\n                        'transactions',\n                        'transactions/{transaction}',\n                    ],\n                    'post': [\n                        'transfer_to_trading',\n                        'transfer_to_main',\n                        'address/{currency}',\n                        'payout',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'maker': -0.01 / 100,\n                    'taker': 0.1 / 100,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.00085,\n                        'BCC': 0.0018,\n                        'ETH': 0.00215,\n                        'BCH': 0.0018,\n                        'USDT': 100,\n                        'DASH': 0.03,\n                        'BTG': 0.0005,\n                        'LTC': 0.003,\n                        'ZEC': 0.0001,\n                        'XMR': 0.09,\n                        '1ST': 0.84,\n                        'ADX': 5.7,\n                        'AE': 6.7,\n                        'AEON': 0.01006,\n                        'AIR': 565,\n                        'AMP': 9,\n                        'ANT': 6.7,\n                        'ARDR': 1,\n                        'ARN': 18.5,\n                        'ART': 26,\n                        'ATB': 0.0004,\n                        'ATL': 27,\n                        'ATM': 504,\n                        'ATS': 860,\n                        'AVT': 1.9,\n                        'BAS': 113,\n                        'BCN': 0.1,\n                        'BET': 124,\n                        'BKB': 46,\n                        'BMC': 32,\n                        'BMT': 100,\n                        'BNT': 2.57,\n                        'BQX': 4.7,\n                        'BTM': 40,\n                        'BTX': 0.04,\n                        'BUS': 0.004,\n                        'CCT': 115,\n                        'CDT': 100,\n                        'CDX': 30,\n                        'CFI': 61,\n                        'CLD': 0.88,\n                        'CND': 574,\n                        'CNX': 0.04,\n                        'COSS': 65,\n                        'CSNO': 16,\n                        'CTR': 15,\n                        'CTX': 146,\n                        'CVC': 8.46,\n                        'DBIX': 0.0168,\n                        'DCN': 120000,\n                        'DCT': 0.02,\n                        'DDF': 342,\n                        'DENT': 6240,\n                        'DGB': 0.4,\n                        'DGD': 0.01,\n                        'DICE': 0.32,\n                        'DLT': 0.26,\n                        'DNT': 0.21,\n                        'DOGE': 2,\n                        'DOV': 34,\n                        'DRPU': 24,\n                        'DRT': 240,\n                        'DSH': 0.017,\n                        'EBET': 84,\n                        'EBTC': 20,\n                        'EBTCOLD': 6.6,\n                        'ECAT': 14,\n                        'EDG': 2,\n                        'EDO': 2.9,\n                        'ELE': 0.00172,\n                        'ELM': 0.004,\n                        'EMC': 0.03,\n                        'EMGO': 14,\n                        'ENJ': 163,\n                        'EOS': 1.5,\n                        'ERO': 34,\n                        'ETBS': 15,\n                        'ETC': 0.002,\n                        'ETP': 0.004,\n                        'EVX': 5.4,\n                        'EXN': 456,\n                        'FRD': 65,\n                        'FUEL': 123.00105,\n                        'FUN': 202.9598309,\n                        'FYN': 1.849,\n                        'FYP': 66.13,\n                        'GNO': 0.0034,\n                        'GUP': 4,\n                        'GVT': 1.2,\n                        'HAC': 144,\n                        'HDG': 7,\n                        'HGT': 1082,\n                        'HPC': 0.4,\n                        'HVN': 120,\n                        'ICN': 0.55,\n                        'ICO': 34,\n                        'ICOS': 0.35,\n                        'IND': 76,\n                        'INDI': 5913,\n                        'ITS': 15.0012,\n                        'IXT': 11,\n                        'KBR': 143,\n                        'KICK': 112,\n                        'LA': 41,\n                        'LAT': 1.44,\n                        'LIFE': 13000,\n                        'LRC': 27,\n                        'LSK': 0.3,\n                        'LUN': 0.34,\n                        'MAID': 5,\n                        'MANA': 143,\n                        'MCAP': 5.44,\n                        'MIPS': 43,\n                        'MNE': 1.33,\n                        'MSP': 121,\n                        'MTH': 92,\n                        'MYB': 3.9,\n                        'NDC': 165,\n                        'NEBL': 0.04,\n                        'NET': 3.96,\n                        'NTO': 998,\n                        'NXC': 13.39,\n                        'NXT': 3,\n                        'OAX': 15,\n                        'ODN': 0.004,\n                        'OMG': 2,\n                        'OPT': 335,\n                        'ORME': 2.8,\n                        'OTN': 0.57,\n                        'PAY': 3.1,\n                        'PIX': 96,\n                        'PLBT': 0.33,\n                        'PLR': 114,\n                        'PLU': 0.87,\n                        'POE': 784,\n                        'POLL': 3.5,\n                        'PPT': 2,\n                        'PRE': 32,\n                        'PRG': 39,\n                        'PRO': 41,\n                        'PRS': 60,\n                        'PTOY': 0.5,\n                        'QAU': 63,\n                        'QCN': 0.03,\n                        'QTUM': 0.04,\n                        'QVT': 64,\n                        'REP': 0.02,\n                        'RKC': 15,\n                        'RVT': 14,\n                        'SAN': 2.24,\n                        'SBD': 0.03,\n                        'SCL': 2.6,\n                        'SISA': 1640,\n                        'SKIN': 407,\n                        'SMART': 0.4,\n                        'SMS': 0.0375,\n                        'SNC': 36,\n                        'SNGLS': 4,\n                        'SNM': 48,\n                        'SNT': 233,\n                        'STEEM': 0.01,\n                        'STRAT': 0.01,\n                        'STU': 14,\n                        'STX': 11,\n                        'SUB': 17,\n                        'SUR': 3,\n                        'SWT': 0.51,\n                        'TAAS': 0.91,\n                        'TBT': 2.37,\n                        'TFL': 15,\n                        'TIME': 0.03,\n                        'TIX': 7.1,\n                        'TKN': 1,\n                        'TKR': 84,\n                        'TNT': 90,\n                        'TRST': 1.6,\n                        'TRX': 1395,\n                        'UET': 480,\n                        'UGT': 15,\n                        'VEN': 14,\n                        'VERI': 0.037,\n                        'VIB': 50,\n                        'VIBE': 145,\n                        'VOISE': 618,\n                        'WEALTH': 0.0168,\n                        'WINGS': 2.4,\n                        'WTC': 0.75,\n                        'XAUR': 3.23,\n                        'XDN': 0.01,\n                        'XEM': 15,\n                        'XUC': 0.9,\n                        'YOYOW': 140,\n                        'ZAP': 24,\n                        'ZRX': 23,\n                        'ZSC': 191,\n                    },\n                    'deposit': {\n                        'BTC': 0.0006,\n                        'ETH': 0.003,\n                        'BCH': 0,\n                        'USDT': 0,\n                        'BTG': 0,\n                        'LTC': 0,\n                        'ZEC': 0,\n                        'XMR': 0,\n                        '1ST': 0,\n                        'ADX': 0,\n                        'AE': 0,\n                        'AEON': 0,\n                        'AIR': 0,\n                        'AMP': 0,\n                        'ANT': 0,\n                        'ARDR': 0,\n                        'ARN': 0,\n                        'ART': 0,\n                        'ATB': 0,\n                        'ATL': 0,\n                        'ATM': 0,\n                        'ATS': 0,\n                        'AVT': 0,\n                        'BAS': 0,\n                        'BCN': 0,\n                        'BET': 0,\n                        'BKB': 0,\n                        'BMC': 0,\n                        'BMT': 0,\n                        'BNT': 0,\n                        'BQX': 0,\n                        'BTM': 0,\n                        'BTX': 0,\n                        'BUS': 0,\n                        'CCT': 0,\n                        'CDT': 0,\n                        'CDX': 0,\n                        'CFI': 0,\n                        'CLD': 0,\n                        'CND': 0,\n                        'CNX': 0,\n                        'COSS': 0,\n                        'CSNO': 0,\n                        'CTR': 0,\n                        'CTX': 0,\n                        'CVC': 0,\n                        'DBIX': 0,\n                        'DCN': 0,\n                        'DCT': 0,\n                        'DDF': 0,\n                        'DENT': 0,\n                        'DGB': 0,\n                        'DGD': 0,\n                        'DICE': 0,\n                        'DLT': 0,\n                        'DNT': 0,\n                        'DOGE': 0,\n                        'DOV': 0,\n                        'DRPU': 0,\n                        'DRT': 0,\n                        'DSH': 0,\n                        'EBET': 0,\n                        'EBTC': 0,\n                        'EBTCOLD': 0,\n                        'ECAT': 0,\n                        'EDG': 0,\n                        'EDO': 0,\n                        'ELE': 0,\n                        'ELM': 0,\n                        'EMC': 0,\n                        'EMGO': 0,\n                        'ENJ': 0,\n                        'EOS': 0,\n                        'ERO': 0,\n                        'ETBS': 0,\n                        'ETC': 0,\n                        'ETP': 0,\n                        'EVX': 0,\n                        'EXN': 0,\n                        'FRD': 0,\n                        'FUEL': 0,\n                        'FUN': 0,\n                        'FYN': 0,\n                        'FYP': 0,\n                        'GNO': 0,\n                        'GUP': 0,\n                        'GVT': 0,\n                        'HAC': 0,\n                        'HDG': 0,\n                        'HGT': 0,\n                        'HPC': 0,\n                        'HVN': 0,\n                        'ICN': 0,\n                        'ICO': 0,\n                        'ICOS': 0,\n                        'IND': 0,\n                        'INDI': 0,\n                        'ITS': 0,\n                        'IXT': 0,\n                        'KBR': 0,\n                        'KICK': 0,\n                        'LA': 0,\n                        'LAT': 0,\n                        'LIFE': 0,\n                        'LRC': 0,\n                        'LSK': 0,\n                        'LUN': 0,\n                        'MAID': 0,\n                        'MANA': 0,\n                        'MCAP': 0,\n                        'MIPS': 0,\n                        'MNE': 0,\n                        'MSP': 0,\n                        'MTH': 0,\n                        'MYB': 0,\n                        'NDC': 0,\n                        'NEBL': 0,\n                        'NET': 0,\n                        'NTO': 0,\n                        'NXC': 0,\n                        'NXT': 0,\n                        'OAX': 0,\n                        'ODN': 0,\n                        'OMG': 0,\n                        'OPT': 0,\n                        'ORME': 0,\n                        'OTN': 0,\n                        'PAY': 0,\n                        'PIX': 0,\n                        'PLBT': 0,\n                        'PLR': 0,\n                        'PLU': 0,\n                        'POE': 0,\n                        'POLL': 0,\n                        'PPT': 0,\n                        'PRE': 0,\n                        'PRG': 0,\n                        'PRO': 0,\n                        'PRS': 0,\n                        'PTOY': 0,\n                        'QAU': 0,\n                        'QCN': 0,\n                        'QTUM': 0,\n                        'QVT': 0,\n                        'REP': 0,\n                        'RKC': 0,\n                        'RVT': 0,\n                        'SAN': 0,\n                        'SBD': 0,\n                        'SCL': 0,\n                        'SISA': 0,\n                        'SKIN': 0,\n                        'SMART': 0,\n                        'SMS': 0,\n                        'SNC': 0,\n                        'SNGLS': 0,\n                        'SNM': 0,\n                        'SNT': 0,\n                        'STEEM': 0,\n                        'STRAT': 0,\n                        'STU': 0,\n                        'STX': 0,\n                        'SUB': 0,\n                        'SUR': 0,\n                        'SWT': 0,\n                        'TAAS': 0,\n                        'TBT': 0,\n                        'TFL': 0,\n                        'TIME': 0,\n                        'TIX': 0,\n                        'TKN': 0,\n                        'TKR': 0,\n                        'TNT': 0,\n                        'TRST': 0,\n                        'TRX': 0,\n                        'UET': 0,\n                        'UGT': 0,\n                        'VEN': 0,\n                        'VERI': 0,\n                        'VIB': 0,\n                        'VIBE': 0,\n                        'VOISE': 0,\n                        'WEALTH': 0,\n                        'WINGS': 0,\n                        'WTC': 0,\n                        'XAUR': 0,\n                        'XDN': 0,\n                        'XEM': 0,\n                        'XUC': 0,\n                        'YOYOW': 0,\n                        'ZAP': 0,\n                        'ZRX': 0,\n                        'ZSC': 0,\n                    },\n                },\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        if (currency === 'XBT')\n            return 'BTC';\n        if (currency === 'DRK')\n            return 'DASH';\n        if (currency === 'CAT')\n            return 'BitClave';\n        if (currency === 'USD')\n            return 'USDT';\n        return currency;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetSymbols ();\n        let result = [];\n        for (let p = 0; p < markets['symbols'].length; p++) {\n            let market = markets['symbols'][p];\n            let id = market['symbol'];\n            let baseId = market['commodity'];\n            let quoteId = market['currency'];\n            let lot = parseFloat (market['lot']);\n            let step = parseFloat (market['step']);\n            let base = this.commonCurrencyCode (baseId);\n            let quote = this.commonCurrencyCode (quoteId);\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'baseId': baseId,\n                'quoteId': quoteId,\n                'lot': lot,\n                'step': step,\n                'info': market,\n                'precision': {\n                    'amount': this.precisionFromString (market['lot']),\n                    'price': this.precisionFromString (market['step']),\n                },\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': step,\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                },\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let method = this.safeString (params, 'type', 'trading');\n        method += 'GetBalance';\n        let query = this.omit (params, 'type');\n        let response = await this[method] (query);\n        let balances = response['balance'];\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let code = balance['currency_code'];\n            let currency = this.commonCurrencyCode (code);\n            let free = this.safeFloat (balance, 'cash', 0.0);\n            free = this.safeFloat (balance, 'balance', free);\n            let used = this.safeFloat (balance, 'reserved', 0.0);\n            let account = {\n                'free': free,\n                'used': used,\n                'total': this.sum (free, used),\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetSymbolOrderbook (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['timestamp'];\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'bid'),\n            'ask': this.safeFloat (ticker, 'ask'),\n            'vwap': undefined,\n            'open': this.safeFloat (ticker, 'open'),\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'volume'),\n            'quoteVolume': this.safeFloat (ticker, 'volume_quote'),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTicker (params);\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetSymbolTicker (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        if ('message' in ticker)\n            throw new ExchangeError (this.id + ' ' + ticker['message']);\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        return {\n            'info': trade,\n            'id': trade[0],\n            'timestamp': trade[3],\n            'datetime': this.iso8601 (trade[3]),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade[4],\n            'price': parseFloat (trade[1]),\n            'amount': parseFloat (trade[2]),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetSymbolTrades (this.extend ({\n            'symbol': market['id'],\n            // 'from': 0,\n            // 'till': 100,\n            // 'by': 'ts', // or by trade_id\n            // 'sort': 'desc', // or asc\n            // 'start_index': 0,\n            // 'max_results': 1000,\n            // 'format_item': 'object',\n            // 'format_price': 'number',\n            // 'format_amount': 'number',\n            // 'format_tid': 'string',\n            // 'format_timestamp': 'millisecond',\n            // 'format_wrap': false,\n            'side': 'true',\n        }, params));\n        return this.parseTrades (response['trades'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        // check if amount can be evenly divided into lots\n        // they want integer quantity in lot units\n        let quantity = parseFloat (amount) / market['lot'];\n        let wholeLots = Math.round (quantity);\n        let difference = quantity - wholeLots;\n        if (Math.abs (difference) > market['step'])\n            throw new ExchangeError (this.id + ' order amount should be evenly divisible by lot unit size of ' + market['lot'].toString ());\n        let clientOrderId = this.milliseconds ();\n        let order = {\n            'clientOrderId': clientOrderId.toString (),\n            'symbol': market['id'],\n            'side': side,\n            'quantity': wholeLots.toString (), // quantity in integer lot units\n            'type': type,\n        };\n        if (type === 'limit') {\n            order['price'] = this.priceToPrecision (symbol, price);\n        } else {\n            order['timeInForce'] = 'FOK';\n        }\n        let response = await this.tradingPostNewOrder (this.extend (order, params));\n        return this.parseOrder (response['ExecutionReport'], market);\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.tradingPostCancelOrder (this.extend ({\n            'clientOrderId': id,\n        }, params));\n    }\n\n    parseOrderStatus (status) {\n        let statuses = {\n            'new': 'open',\n            'partiallyFilled': 'open',\n            'filled': 'closed',\n            'canceled': 'canceled',\n            'rejected': 'rejected',\n            'expired': 'expired',\n        };\n        return this.safeString (statuses, status);\n    }\n\n    parseOrder (order, market = undefined) {\n        let timestamp = this.safeInteger (order, 'lastTimestamp');\n        if (typeof timestamp === 'undefined')\n            timestamp = this.safeInteger (order, 'timestamp');\n        let symbol = undefined;\n        if (!market)\n            market = this.markets_by_id[order['symbol']];\n        let status = this.safeString (order, 'orderStatus');\n        if (status)\n            status = this.parseOrderStatus (status);\n        let averagePrice = this.safeFloat (order, 'avgPrice', 0.0);\n        let price = this.safeFloat (order, 'orderPrice');\n        if (typeof price === 'undefined')\n            price = this.safeFloat (order, 'price');\n        let amount = this.safeFloat (order, 'orderQuantity');\n        if (typeof amount === 'undefined')\n            amount = this.safeFloat (order, 'quantity');\n        let remaining = this.safeFloat (order, 'quantityLeaves');\n        if (!remaining)\n            remaining = this.safeFloat (order, 'leavesQuantity');\n        let filled = undefined;\n        let cost = undefined;\n        let amountDefined = (typeof amount !== 'undefined');\n        let remainingDefined = (typeof remaining !== 'undefined');\n        if (market) {\n            symbol = market['symbol'];\n            if (amountDefined)\n                amount *= market['lot'];\n            if (remainingDefined)\n                remaining *= market['lot'];\n        }\n        if (amountDefined) {\n            if (remainingDefined) {\n                filled = amount - remaining;\n                cost = averagePrice * filled;\n            }\n        }\n        return {\n            'id': order['clientOrderId'].toString (),\n            'info': order,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': status,\n            'symbol': symbol,\n            'type': order['type'],\n            'side': order['side'],\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'fee': undefined,\n        };\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.tradingGetOrder (this.extend ({\n            'clientOrderId': id,\n        }, params));\n        if (response['orders'][0]) {\n            return this.parseOrder (response['orders'][0]);\n        }\n        throw new OrderNotFound (this.id + ' fetchOrder() error: ' + this.response);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let statuses = [ 'new', 'partiallyFiiled' ];\n        let market = undefined;\n        let request = {\n            'sort': 'desc',\n            'statuses': statuses.join (','),\n        };\n        if (symbol) {\n            market = this.market (symbol);\n            request['symbols'] = market['id'];\n        }\n        let response = await this.tradingGetOrdersActive (this.extend (request, params));\n        return this.parseOrders (response['orders'], market, since, limit);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        let statuses = [ 'filled', 'canceled', 'rejected', 'expired' ];\n        let request = {\n            'sort': 'desc',\n            'statuses': statuses.join (','),\n            'max_results': 1000,\n        };\n        if (symbol) {\n            market = this.market (symbol);\n            request['symbols'] = market['id'];\n        }\n        let response = await this.tradingGetOrdersRecent (this.extend (request, params));\n        return this.parseOrders (response['orders'], market, since, limit);\n    }\n\n    async withdraw (code, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let currency = this.currency (code);\n        let request = {\n            'currency_code': currency['id'],\n            'amount': amount,\n            'address': address,\n        };\n        if (tag)\n            request['paymentId'] = tag;\n        let response = await this.paymentPostPayout (this.extend (request, params));\n        return {\n            'info': response,\n            'id': response['transaction'],\n        };\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = '/' + 'api' + '/' + this.version + '/' + api + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let payload = { 'nonce': nonce, 'apikey': this.apiKey };\n            query = this.extend (payload, query);\n            if (method === 'GET')\n                url += '?' + this.urlencode (query);\n            else\n                url += '?' + this.urlencode (payload);\n            let auth = url;\n            if (method === 'POST') {\n                if (Object.keys (query).length) {\n                    body = this.urlencode (query);\n                    auth += body;\n                }\n            }\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'X-Signature': this.hmac (this.encode (auth), this.encode (this.secret), 'sha512').toLowerCase (),\n            };\n        }\n        url = this.urls['api'] + url;\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('code' in response) {\n            if ('ExecutionReport' in response) {\n                if (response['ExecutionReport']['orderRejectReason'] === 'orderExceedsLimit')\n                    throw new InsufficientFunds (this.id + ' ' + this.json (response));\n            }\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n        return response;\n    }\n};\n","'use strict';\n\n// ---------------------------------------------------------------------------\n\nconst hitbtc = require ('./hitbtc');\nconst { ExchangeError, OrderNotFound, InsufficientFunds, InvalidOrder } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class hitbtc2 extends hitbtc {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'hitbtc2',\n            'name': 'HitBTC v2',\n            'countries': 'UK',\n            'rateLimit': 1500,\n            'version': '2',\n            'has': {\n                'CORS': true,\n                'fetchCurrencies': true,\n                'fetchOHLCV': true,\n                'fetchTickers': true,\n                'fetchOrder': true,\n                'fetchOrders': false,\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': true,\n                'fetchMyTrades': true,\n                'withdraw': true,\n            },\n            'timeframes': {\n                '1m': 'M1',\n                '3m': 'M3',\n                '5m': 'M5',\n                '15m': 'M15',\n                '30m': 'M30', // default\n                '1h': 'H1',\n                '4h': 'H4',\n                '1d': 'D1',\n                '1w': 'D7',\n                '1M': '1M',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766555-8eaec20e-5edc-11e7-9c5b-6dc69fc42f5e.jpg',\n                'api': 'https://api.hitbtc.com',\n                'www': 'https://hitbtc.com',\n                'doc': 'https://api.hitbtc.com',\n                'fees': [\n                    'https://hitbtc.com/fees-and-limits',\n                    'https://support.hitbtc.com/hc/en-us/articles/115005148605-Fees-and-limits',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'symbol', // Available Currency Symbols\n                        'symbol/{symbol}', // Get symbol info\n                        'currency', // Available Currencies\n                        'currency/{currency}', // Get currency info\n                        'ticker', // Ticker list for all symbols\n                        'ticker/{symbol}', // Ticker for symbol\n                        'trades/{symbol}', // Trades\n                        'orderbook/{symbol}', // Orderbook\n                        'candles/{symbol}', // Candles\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'order', // List your current open orders\n                        'order/{clientOrderId}', // Get a single order by clientOrderId\n                        'trading/balance', // Get trading balance\n                        'trading/fee/{symbol}', // Get trading fee rate\n                        'history/trades', // Get historical trades\n                        'history/order', // Get historical orders\n                        'history/order/{id}/trades', // Get historical trades by specified order\n                        'account/balance', // Get main acccount balance\n                        'account/transactions', // Get account transactions\n                        'account/transactions/{id}', // Get account transaction by id\n                        'account/crypto/address/{currency}', // Get deposit crypro address\n                    ],\n                    'post': [\n                        'order', // Create new order\n                        'account/crypto/withdraw', // Withdraw crypro\n                        'account/crypto/address/{currency}', // Create new deposit crypro address\n                        'account/transfer', // Transfer amount to trading\n                    ],\n                    'put': [\n                        'order/{clientOrderId}', // Create new order\n                        'account/crypto/withdraw/{id}', // Commit withdraw crypro\n                    ],\n                    'delete': [\n                        'order', // Cancel all open orders\n                        'order/{clientOrderId}', // Cancel order\n                        'account/crypto/withdraw/{id}', // Rollback withdraw crypro\n                    ],\n                    'patch': [\n                        'order/{clientOrderId}', // Cancel Replace order\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'maker': -0.01 / 100,\n                    'taker': 0.1 / 100,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.00085,\n                        'BCC': 0.0018,\n                        'ETH': 0.00215,\n                        'BCH': 0.0018,\n                        'USDT': 100,\n                        'DASH': 0.03,\n                        'BTG': 0.0005,\n                        'LTC': 0.003,\n                        'ZEC': 0.0001,\n                        'XMR': 0.09,\n                        '1ST': 0.84,\n                        'ADX': 5.7,\n                        'AE': 6.7,\n                        'AEON': 0.01006,\n                        'AIR': 565,\n                        'AMP': 9,\n                        'ANT': 6.7,\n                        'ARDR': 1,\n                        'ARN': 18.5,\n                        'ART': 26,\n                        'ATB': 0.0004,\n                        'ATL': 27,\n                        'ATM': 504,\n                        'ATS': 860,\n                        'AVT': 1.9,\n                        'BAS': 113,\n                        'BCN': 0.1,\n                        'BET': 124,\n                        'BKB': 46,\n                        'BMC': 32,\n                        'BMT': 100,\n                        'BNT': 2.57,\n                        'BQX': 4.7,\n                        'BTM': 40,\n                        'BTX': 0.04,\n                        'BUS': 0.004,\n                        'CCT': 115,\n                        'CDT': 100,\n                        'CDX': 30,\n                        'CFI': 61,\n                        'CLD': 0.88,\n                        'CND': 574,\n                        'CNX': 0.04,\n                        'COSS': 65,\n                        'CSNO': 16,\n                        'CTR': 15,\n                        'CTX': 146,\n                        'CVC': 8.46,\n                        'DBIX': 0.0168,\n                        'DCN': 120000,\n                        'DCT': 0.02,\n                        'DDF': 342,\n                        'DENT': 6240,\n                        'DGB': 0.4,\n                        'DGD': 0.01,\n                        'DICE': 0.32,\n                        'DLT': 0.26,\n                        'DNT': 0.21,\n                        'DOGE': 2,\n                        'DOV': 34,\n                        'DRPU': 24,\n                        'DRT': 240,\n                        'DSH': 0.017,\n                        'EBET': 84,\n                        'EBTC': 20,\n                        'EBTCOLD': 6.6,\n                        'ECAT': 14,\n                        'EDG': 2,\n                        'EDO': 2.9,\n                        'ELE': 0.00172,\n                        'ELM': 0.004,\n                        'EMC': 0.03,\n                        'EMGO': 14,\n                        'ENJ': 163,\n                        'EOS': 1.5,\n                        'ERO': 34,\n                        'ETBS': 15,\n                        'ETC': 0.002,\n                        'ETP': 0.004,\n                        'EVX': 5.4,\n                        'EXN': 456,\n                        'FRD': 65,\n                        'FUEL': 123.00105,\n                        'FUN': 202.9598309,\n                        'FYN': 1.849,\n                        'FYP': 66.13,\n                        'GNO': 0.0034,\n                        'GUP': 4,\n                        'GVT': 1.2,\n                        'HAC': 144,\n                        'HDG': 7,\n                        'HGT': 1082,\n                        'HPC': 0.4,\n                        'HVN': 120,\n                        'ICN': 0.55,\n                        'ICO': 34,\n                        'ICOS': 0.35,\n                        'IND': 76,\n                        'INDI': 5913,\n                        'ITS': 15.0012,\n                        'IXT': 11,\n                        'KBR': 143,\n                        'KICK': 112,\n                        'LA': 41,\n                        'LAT': 1.44,\n                        'LIFE': 13000,\n                        'LRC': 27,\n                        'LSK': 0.3,\n                        'LUN': 0.34,\n                        'MAID': 5,\n                        'MANA': 143,\n                        'MCAP': 5.44,\n                        'MIPS': 43,\n                        'MNE': 1.33,\n                        'MSP': 121,\n                        'MTH': 92,\n                        'MYB': 3.9,\n                        'NDC': 165,\n                        'NEBL': 0.04,\n                        'NET': 3.96,\n                        'NTO': 998,\n                        'NXC': 13.39,\n                        'NXT': 3,\n                        'OAX': 15,\n                        'ODN': 0.004,\n                        'OMG': 2,\n                        'OPT': 335,\n                        'ORME': 2.8,\n                        'OTN': 0.57,\n                        'PAY': 3.1,\n                        'PIX': 96,\n                        'PLBT': 0.33,\n                        'PLR': 114,\n                        'PLU': 0.87,\n                        'POE': 784,\n                        'POLL': 3.5,\n                        'PPT': 2,\n                        'PRE': 32,\n                        'PRG': 39,\n                        'PRO': 41,\n                        'PRS': 60,\n                        'PTOY': 0.5,\n                        'QAU': 63,\n                        'QCN': 0.03,\n                        'QTUM': 0.04,\n                        'QVT': 64,\n                        'REP': 0.02,\n                        'RKC': 15,\n                        'RVT': 14,\n                        'SAN': 2.24,\n                        'SBD': 0.03,\n                        'SCL': 2.6,\n                        'SISA': 1640,\n                        'SKIN': 407,\n                        'SMART': 0.4,\n                        'SMS': 0.0375,\n                        'SNC': 36,\n                        'SNGLS': 4,\n                        'SNM': 48,\n                        'SNT': 233,\n                        'STEEM': 0.01,\n                        'STRAT': 0.01,\n                        'STU': 14,\n                        'STX': 11,\n                        'SUB': 17,\n                        'SUR': 3,\n                        'SWT': 0.51,\n                        'TAAS': 0.91,\n                        'TBT': 2.37,\n                        'TFL': 15,\n                        'TIME': 0.03,\n                        'TIX': 7.1,\n                        'TKN': 1,\n                        'TKR': 84,\n                        'TNT': 90,\n                        'TRST': 1.6,\n                        'TRX': 1395,\n                        'UET': 480,\n                        'UGT': 15,\n                        'VEN': 14,\n                        'VERI': 0.037,\n                        'VIB': 50,\n                        'VIBE': 145,\n                        'VOISE': 618,\n                        'WEALTH': 0.0168,\n                        'WINGS': 2.4,\n                        'WTC': 0.75,\n                        'XAUR': 3.23,\n                        'XDN': 0.01,\n                        'XEM': 15,\n                        'XUC': 0.9,\n                        'YOYOW': 140,\n                        'ZAP': 24,\n                        'ZRX': 23,\n                        'ZSC': 191,\n                    },\n                    'deposit': {\n                        'BTC': 0.0006,\n                        'ETH': 0.003,\n                        'BCH': 0,\n                        'USDT': 0,\n                        'BTG': 0,\n                        'LTC': 0,\n                        'ZEC': 0,\n                        'XMR': 0,\n                        '1ST': 0,\n                        'ADX': 0,\n                        'AE': 0,\n                        'AEON': 0,\n                        'AIR': 0,\n                        'AMP': 0,\n                        'ANT': 0,\n                        'ARDR': 0,\n                        'ARN': 0,\n                        'ART': 0,\n                        'ATB': 0,\n                        'ATL': 0,\n                        'ATM': 0,\n                        'ATS': 0,\n                        'AVT': 0,\n                        'BAS': 0,\n                        'BCN': 0,\n                        'BET': 0,\n                        'BKB': 0,\n                        'BMC': 0,\n                        'BMT': 0,\n                        'BNT': 0,\n                        'BQX': 0,\n                        'BTM': 0,\n                        'BTX': 0,\n                        'BUS': 0,\n                        'CCT': 0,\n                        'CDT': 0,\n                        'CDX': 0,\n                        'CFI': 0,\n                        'CLD': 0,\n                        'CND': 0,\n                        'CNX': 0,\n                        'COSS': 0,\n                        'CSNO': 0,\n                        'CTR': 0,\n                        'CTX': 0,\n                        'CVC': 0,\n                        'DBIX': 0,\n                        'DCN': 0,\n                        'DCT': 0,\n                        'DDF': 0,\n                        'DENT': 0,\n                        'DGB': 0,\n                        'DGD': 0,\n                        'DICE': 0,\n                        'DLT': 0,\n                        'DNT': 0,\n                        'DOGE': 0,\n                        'DOV': 0,\n                        'DRPU': 0,\n                        'DRT': 0,\n                        'DSH': 0,\n                        'EBET': 0,\n                        'EBTC': 0,\n                        'EBTCOLD': 0,\n                        'ECAT': 0,\n                        'EDG': 0,\n                        'EDO': 0,\n                        'ELE': 0,\n                        'ELM': 0,\n                        'EMC': 0,\n                        'EMGO': 0,\n                        'ENJ': 0,\n                        'EOS': 0,\n                        'ERO': 0,\n                        'ETBS': 0,\n                        'ETC': 0,\n                        'ETP': 0,\n                        'EVX': 0,\n                        'EXN': 0,\n                        'FRD': 0,\n                        'FUEL': 0,\n                        'FUN': 0,\n                        'FYN': 0,\n                        'FYP': 0,\n                        'GNO': 0,\n                        'GUP': 0,\n                        'GVT': 0,\n                        'HAC': 0,\n                        'HDG': 0,\n                        'HGT': 0,\n                        'HPC': 0,\n                        'HVN': 0,\n                        'ICN': 0,\n                        'ICO': 0,\n                        'ICOS': 0,\n                        'IND': 0,\n                        'INDI': 0,\n                        'ITS': 0,\n                        'IXT': 0,\n                        'KBR': 0,\n                        'KICK': 0,\n                        'LA': 0,\n                        'LAT': 0,\n                        'LIFE': 0,\n                        'LRC': 0,\n                        'LSK': 0,\n                        'LUN': 0,\n                        'MAID': 0,\n                        'MANA': 0,\n                        'MCAP': 0,\n                        'MIPS': 0,\n                        'MNE': 0,\n                        'MSP': 0,\n                        'MTH': 0,\n                        'MYB': 0,\n                        'NDC': 0,\n                        'NEBL': 0,\n                        'NET': 0,\n                        'NTO': 0,\n                        'NXC': 0,\n                        'NXT': 0,\n                        'OAX': 0,\n                        'ODN': 0,\n                        'OMG': 0,\n                        'OPT': 0,\n                        'ORME': 0,\n                        'OTN': 0,\n                        'PAY': 0,\n                        'PIX': 0,\n                        'PLBT': 0,\n                        'PLR': 0,\n                        'PLU': 0,\n                        'POE': 0,\n                        'POLL': 0,\n                        'PPT': 0,\n                        'PRE': 0,\n                        'PRG': 0,\n                        'PRO': 0,\n                        'PRS': 0,\n                        'PTOY': 0,\n                        'QAU': 0,\n                        'QCN': 0,\n                        'QTUM': 0,\n                        'QVT': 0,\n                        'REP': 0,\n                        'RKC': 0,\n                        'RVT': 0,\n                        'SAN': 0,\n                        'SBD': 0,\n                        'SCL': 0,\n                        'SISA': 0,\n                        'SKIN': 0,\n                        'SMART': 0,\n                        'SMS': 0,\n                        'SNC': 0,\n                        'SNGLS': 0,\n                        'SNM': 0,\n                        'SNT': 0,\n                        'STEEM': 0,\n                        'STRAT': 0,\n                        'STU': 0,\n                        'STX': 0,\n                        'SUB': 0,\n                        'SUR': 0,\n                        'SWT': 0,\n                        'TAAS': 0,\n                        'TBT': 0,\n                        'TFL': 0,\n                        'TIME': 0,\n                        'TIX': 0,\n                        'TKN': 0,\n                        'TKR': 0,\n                        'TNT': 0,\n                        'TRST': 0,\n                        'TRX': 0,\n                        'UET': 0,\n                        'UGT': 0,\n                        'VEN': 0,\n                        'VERI': 0,\n                        'VIB': 0,\n                        'VIBE': 0,\n                        'VOISE': 0,\n                        'WEALTH': 0,\n                        'WINGS': 0,\n                        'WTC': 0,\n                        'XAUR': 0,\n                        'XDN': 0,\n                        'XEM': 0,\n                        'XUC': 0,\n                        'YOYOW': 0,\n                        'ZAP': 0,\n                        'ZRX': 0,\n                        'ZSC': 0,\n                    },\n                },\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        let currencies = {\n            'XBT': 'BTC',\n            'DRK': 'DASH',\n            'CAT': 'BitClave',\n            'USD': 'USDT',\n        };\n        if (currency in currencies)\n            return currencies[currency];\n        return currency;\n    }\n\n    feeToPrecision (symbol, fee) {\n        return this.truncate (fee, 8);\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetSymbol ();\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let id = market['id'];\n            let baseId = market['baseCurrency'];\n            let quoteId = market['quoteCurrency'];\n            let base = this.commonCurrencyCode (baseId);\n            let quote = this.commonCurrencyCode (quoteId);\n            let symbol = base + '/' + quote;\n            let lot = parseFloat (market['quantityIncrement']);\n            let step = parseFloat (market['tickSize']);\n            let precision = {\n                'price': this.precisionFromString (market['tickSize']),\n                'amount': this.precisionFromString (market['quantityIncrement']),\n            };\n            let taker = parseFloat (market['takeLiquidityRate']);\n            let maker = parseFloat (market['provideLiquidityRate']);\n            result.push (this.extend (this.fees['trading'], {\n                'info': market,\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'baseId': baseId,\n                'quoteId': quoteId,\n                'active': true,\n                'lot': lot,\n                'step': step,\n                'taker': taker,\n                'maker': maker,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': step,\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': lot * step,\n                        'max': undefined,\n                    },\n                },\n            }));\n        }\n        return result;\n    }\n\n    async fetchCurrencies (params = {}) {\n        let currencies = await this.publicGetCurrency (params);\n        let result = {};\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let id = currency['id'];\n            // todo: will need to rethink the fees\n            // to add support for multiple withdrawal/deposit methods and\n            // differentiated fees for each particular method\n            let precision = 8; // default precision, todo: fix \"magic constants\"\n            let code = this.commonCurrencyCode (id);\n            let payin = currency['payinEnabled'];\n            let payout = currency['payoutEnabled'];\n            let transfer = currency['transferEnabled'];\n            let active = payin && payout && transfer;\n            let status = 'ok';\n            if ('disabled' in currency)\n                if (currency['disabled'])\n                    status = 'disabled';\n            let type = (currency['crypto']) ? 'crypto' : 'fiat';\n            result[code] = {\n                'id': id,\n                'code': code,\n                'type': type,\n                'payin': payin,\n                'payout': payout,\n                'transfer': transfer,\n                'info': currency,\n                'name': currency['fullName'],\n                'active': active,\n                'status': status,\n                'fee': undefined, // todo: redesign\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': undefined,\n                        'max': Math.pow (10, precision),\n                    },\n                },\n            };\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let type = this.safeString (params, 'type', 'trading');\n        let method = 'privateGet' + this.capitalize (type) + 'Balance';\n        let balances = await this[method] ();\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let code = balance['currency'];\n            let currency = this.commonCurrencyCode (code);\n            let account = {\n                'free': parseFloat (balance['available']),\n                'used': parseFloat (balance['reserved']),\n                'total': 0.0,\n            };\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1d', since = undefined, limit = undefined) {\n        let timestamp = this.parse8601 (ohlcv['timestamp']);\n        return [\n            timestamp,\n            parseFloat (ohlcv['open']),\n            parseFloat (ohlcv['max']),\n            parseFloat (ohlcv['min']),\n            parseFloat (ohlcv['close']),\n            parseFloat (ohlcv['volumeQuote']),\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n            'period': this.timeframes[timeframe],\n        };\n        if (limit)\n            request['limit'] = limit;\n        let response = await this.publicGetCandlesSymbol (this.extend (request, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderbookSymbol (this.extend ({\n            'symbol': this.marketId (symbol),\n            // 'limit': 100, // default = 100, 0 = unlimited\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'bid', 'ask', 'price', 'size');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.parse8601 (ticker['timestamp']);\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'bid'),\n            'ask': this.safeFloat (ticker, 'ask'),\n            'vwap': undefined,\n            'open': this.safeFloat (ticker, 'open'),\n            'close': this.safeFloat (ticker, 'close'),\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'volume'),\n            'quoteVolume': this.safeFloat (ticker, 'volumeQuote'),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTicker (params);\n        let result = {};\n        for (let i = 0; i < tickers.length; i++) {\n            let ticker = tickers[i];\n            let id = ticker['symbol'];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetTickerSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        if ('message' in ticker)\n            throw new ExchangeError (this.id + ' ' + ticker['message']);\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = this.parse8601 (trade['timestamp']);\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else {\n            let id = trade['symbol'];\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            } else {\n                symbol = id;\n            }\n        }\n        let fee = undefined;\n        if ('fee' in trade) {\n            let currency = market ? market['quote'] : undefined;\n            fee = {\n                'cost': parseFloat (trade['fee']),\n                'currency': currency,\n            };\n        }\n        let orderId = undefined;\n        if ('clientOrderId' in trade)\n            orderId = trade['clientOrderId'];\n        let price = parseFloat (trade['price']);\n        let amount = parseFloat (trade['quantity']);\n        let cost = price * amount;\n        return {\n            'info': trade,\n            'id': trade['id'].toString (),\n            'order': orderId,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': undefined,\n            'side': trade['side'],\n            'price': price,\n            'amount': amount,\n            'cost': cost,\n            'fee': fee,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTradesSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        // their max accepted length is 32 characters\n        let uuid = this.uuid ();\n        let parts = uuid.split ('-');\n        let clientOrderId = parts.join ('');\n        clientOrderId = clientOrderId.slice (0, 32);\n        amount = parseFloat (amount);\n        let request = {\n            'clientOrderId': clientOrderId,\n            'symbol': market['id'],\n            'side': side,\n            'quantity': this.amountToPrecision (symbol, amount),\n            'type': type,\n        };\n        if (type === 'limit') {\n            request['price'] = this.priceToPrecision (symbol, price);\n        } else {\n            request['timeInForce'] = 'FOK';\n        }\n        let response = await this.privatePostOrder (this.extend (request, params));\n        let order = this.parseOrder (response);\n        let id = order['id'];\n        this.orders[id] = order;\n        return order;\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privateDeleteOrderClientOrderId (this.extend ({\n            'clientOrderId': id,\n        }, params));\n    }\n\n    parseOrder (order, market = undefined) {\n        let created = undefined;\n        if ('createdAt' in order)\n            created = this.parse8601 (order['createdAt']);\n        let updated = undefined;\n        if ('updatedAt' in order)\n            updated = this.parse8601 (order['updatedAt']);\n        if (!market)\n            market = this.markets_by_id[order['symbol']];\n        let symbol = market['symbol'];\n        let amount = this.safeFloat (order, 'quantity');\n        let filled = this.safeFloat (order, 'cumQuantity');\n        let status = order['status'];\n        if (status === 'new') {\n            status = 'open';\n        } else if (status === 'suspended') {\n            status = 'open';\n        } else if (status === 'partiallyFilled') {\n            status = 'open';\n        } else if (status === 'filled') {\n            status = 'closed';\n        }\n        let id = order['clientOrderId'].toString ();\n        let price = this.safeFloat (order, 'price');\n        if (typeof price === 'undefined') {\n            if (id in this.orders)\n                price = this.orders[id]['price'];\n        }\n        let remaining = undefined;\n        let cost = undefined;\n        if (typeof amount !== 'undefined') {\n            if (typeof filled !== 'undefined') {\n                remaining = amount - filled;\n                if (typeof price !== 'undefined') {\n                    cost = filled * price;\n                }\n            }\n        }\n        return {\n            'id': id,\n            'timestamp': created,\n            'datetime': this.iso8601 (created),\n            'created': created,\n            'updated': updated,\n            'status': status,\n            'symbol': symbol,\n            'type': order['type'],\n            'side': order['side'],\n            'price': price,\n            'amount': amount,\n            'cost': cost,\n            'filled': filled,\n            'remaining': remaining,\n            'fee': undefined,\n            'info': order,\n        };\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetHistoryOrder (this.extend ({\n            'clientOrderId': id,\n        }, params));\n        let numOrders = response.length;\n        if (numOrders > 0)\n            return this.parseOrder (response[0]);\n        throw new OrderNotFound (this.id + ' order ' + id + ' not found');\n    }\n\n    async fetchOpenOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetOrderClientOrderId (this.extend ({\n            'clientOrderId': id,\n        }, params));\n        return this.parseOrder (response);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        let request = {};\n        if (symbol) {\n            market = this.market (symbol);\n            request['symbol'] = market['id'];\n        }\n        let response = await this.privateGetOrder (this.extend (request, params));\n        return this.parseOrders (response, market, since, limit);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        let request = {};\n        if (symbol) {\n            market = this.market (symbol);\n            request['symbol'] = market['id'];\n        }\n        if (limit)\n            request['limit'] = limit;\n        if (since) {\n            request['from'] = this.iso8601 (since);\n        }\n        let response = await this.privateGetHistoryOrder (this.extend (request, params));\n        return this.parseOrders (response, market, since, limit);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            // 'symbol': 'BTC/USD', // optional\n            // 'sort': 'DESC', // or 'ASC'\n            // 'by': 'timestamp', // or 'id'\tString\ttimestamp by default, or id\n            // 'from':\t'Datetime or Number', // ISO 8601\n            // 'till':\t'Datetime or Number',\n            // 'limit': 100,\n            // 'offset': 0,\n        };\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['symbol'] = market['id'];\n        }\n        if (since)\n            request['from'] = this.iso8601 (since);\n        if (limit)\n            request['limit'] = limit;\n        let response = await this.privateGetHistoryTrades (this.extend (request, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchOrderTrades (id, symbol = undefined, params = {}) {\n        // The id needed here is the exchange's id, and not the clientOrderID, which is\n        // the id that is stored in the unified api order id. In order the get the exchange's id,\n        // you need to grab it from order['info']['id']\n        await this.loadMarkets ();\n        let trades = await this.privateGetHistoryOrderIdTrades (this.extend ({\n            'id': id,\n        }, params));\n        return this.parseTrades (trades);\n    }\n\n    async createDepositAddress (code, params = {}) {\n        await this.loadMarkets ();\n        let currency = this.currency (code);\n        let response = await this.privatePostAccountCryptoAddressCurrency ({\n            'currency': currency['id'],\n        });\n        let address = response['address'];\n        return {\n            'currency': currency,\n            'address': address,\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    async fetchDepositAddress (code, params = {}) {\n        await this.loadMarkets ();\n        let currency = this.currency (code);\n        let response = await this.privateGetAccountCryptoAddressCurrency ({\n            'currency': currency['id'],\n        });\n        let address = response['address'];\n        return {\n            'currency': currency,\n            'address': address,\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    async withdraw (code, amount, address, tag = undefined, params = {}) {\n        let currency = this.currency (code);\n        let request = {\n            'currency': currency['id'],\n            'amount': parseFloat (amount),\n            'address': address,\n        };\n        if (tag)\n            request['paymentId'] = tag;\n        let response = await this.privatePostAccountCryptoWithdraw (this.extend (request, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = '/api' + '/' + this.version + '/';\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            url += api + '/' + this.implodeParams (path, params);\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            url += this.implodeParams (path, params);\n            if (method === 'GET') {\n                if (Object.keys (query).length)\n                    url += '?' + this.urlencode (query);\n            } else {\n                if (Object.keys (query).length)\n                    body = this.json (query);\n            }\n            let payload = this.encode (this.apiKey + ':' + this.secret);\n            let auth = this.stringToBase64 (payload);\n            headers = {\n                'Authorization': 'Basic ' + this.decode (auth),\n                'Content-Type': 'application/json',\n            };\n        }\n        url = this.urls['api'] + url;\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code === 400) {\n            if (body[0] === '{') {\n                let response = JSON.parse (body);\n                if ('error' in response) {\n                    if ('message' in response['error']) {\n                        let message = response['error']['message'];\n                        if (message === 'Order not found') {\n                            throw new OrderNotFound (this.id + ' order not found in active orders');\n                        } else if (message === 'Insufficient funds') {\n                            throw new InsufficientFunds (this.id + ' ' + body);\n                        } else if (message === 'Duplicate clientOrderId') {\n                            throw new InvalidOrder (this.id + ' ' + body);\n                        }\n                    }\n                }\n            }\n            throw new ExchangeError (this.id + ' ' + body);\n        }\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class huobi extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'huobi',\n            'name': 'Huobi',\n            'countries': 'CN',\n            'rateLimit': 2000,\n            'version': 'v3',\n            'has': {\n                'CORS': false,\n                'fetchOHLCV': true,\n            },\n            'timeframes': {\n                '1m': '001',\n                '5m': '005',\n                '15m': '015',\n                '30m': '030',\n                '1h': '060',\n                '1d': '100',\n                '1w': '200',\n                '1M': '300',\n                '1y': '400',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766569-15aa7b9a-5edd-11e7-9e7f-44791f4ee49c.jpg',\n                'api': 'http://api.huobi.com',\n                'www': 'https://www.huobi.com',\n                'doc': 'https://github.com/huobiapi/API_Docs_en/wiki',\n            },\n            'api': {\n                'staticmarket': {\n                    'get': [\n                        '{id}_kline_{period}',\n                        'ticker_{id}',\n                        'depth_{id}',\n                        'depth_{id}_{length}',\n                        'detail_{id}',\n                    ],\n                },\n                'usdmarket': {\n                    'get': [\n                        '{id}_kline_{period}',\n                        'ticker_{id}',\n                        'depth_{id}',\n                        'depth_{id}_{length}',\n                        'detail_{id}',\n                    ],\n                },\n                'trade': {\n                    'post': [\n                        'get_account_info',\n                        'get_orders',\n                        'order_info',\n                        'buy',\n                        'sell',\n                        'buy_market',\n                        'sell_market',\n                        'cancel_order',\n                        'get_new_deal_orders',\n                        'get_order_id_by_trade_id',\n                        'withdraw_coin',\n                        'cancel_withdraw_coin',\n                        'get_withdraw_coin_result',\n                        'transfer',\n                        'loan',\n                        'repayment',\n                        'get_loan_available',\n                        'get_loans',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/CNY': { 'id': 'btc', 'symbol': 'BTC/CNY', 'base': 'BTC', 'quote': 'CNY', 'type': 'staticmarket', 'coinType': 1 },\n                'LTC/CNY': { 'id': 'ltc', 'symbol': 'LTC/CNY', 'base': 'LTC', 'quote': 'CNY', 'type': 'staticmarket', 'coinType': 2 },\n                // 'BTC/USD': { 'id': 'btc', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD', 'type': 'usdmarket',    'coinType': 1 },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let balances = await this.tradePostGetAccountInfo ();\n        let result = { 'info': balances };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let account = this.account ();\n            let available = 'available_' + lowercase + '_display';\n            let frozen = 'frozen_' + lowercase + '_display';\n            let loan = 'loan_' + lowercase + '_display';\n            if (available in balances)\n                account['free'] = parseFloat (balances[available]);\n            if (frozen in balances)\n                account['used'] = parseFloat (balances[frozen]);\n            if (loan in balances)\n                account['used'] = this.sum (account['used'], parseFloat (balances[loan]));\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let market = this.market (symbol);\n        let method = market['type'] + 'GetDepthId';\n        let orderbook = await this[method] (this.extend ({ 'id': market['id'] }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let market = this.market (symbol);\n        let method = market['type'] + 'GetTickerId';\n        let response = await this[method] (this.extend ({\n            'id': market['id'],\n        }, params));\n        let ticker = response['ticker'];\n        let timestamp = parseInt (response['time']) * 1000;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'buy'),\n            'ask': this.safeFloat (ticker, 'sell'),\n            'vwap': undefined,\n            'open': this.safeFloat (ticker, 'open'),\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': this.safeFloat (ticker, 'vol'),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['ts'];\n        return {\n            'info': trade,\n            'id': trade['id'].toString (),\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['direction'],\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let method = market['type'] + 'GetDetailId';\n        let response = await this[method] (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTrades (response['trades'], market, since, limit);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        // not implemented yet\n        return [\n            ohlcv[0],\n            ohlcv[1],\n            ohlcv[2],\n            ohlcv[3],\n            ohlcv[4],\n            ohlcv[6],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let method = market['type'] + 'GetIdKlinePeriod';\n        let ohlcvs = await this[method] (this.extend ({\n            'id': market['id'],\n            'period': this.timeframes[timeframe],\n        }, params));\n        return ohlcvs;\n        // return this.parseOHLCVs (ohlcvs, market, timeframe, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let market = this.market (symbol);\n        let method = 'tradePost' + this.capitalize (side);\n        let order = {\n            'coin_type': market['coinType'],\n            'amount': amount,\n            'market': market['quote'].toLowerCase (),\n        };\n        if (type == 'limit')\n            order['price'] = price;\n        else\n            method += this.capitalize (type);\n        let response = this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.tradePostCancelOrder ({ 'id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'];\n        if (api == 'trade') {\n            this.checkRequiredCredentials ();\n            url += '/api' + this.version;\n            let query = this.keysort (this.extend ({\n                'method': path,\n                'access_key': this.apiKey,\n                'created': this.nonce (),\n            }, params));\n            let queryString = this.urlencode (this.omit (query, 'market'));\n            // secret key must be appended to the query before signing\n            queryString += '&secret_key=' + this.secret;\n            query['sign'] = this.hash (this.encode (queryString));\n            body = this.urlencode (query);\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        } else {\n            url += '/' + api + '/' + this.implodeParams (path, params) + '_json.js';\n            let query = this.omit (params, this.extractParams (path));\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'trade', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('status' in response)\n            if (response['status'] == 'error')\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        if ('code' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst huobipro = require ('./huobipro.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class huobicny extends huobipro {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'huobicny',\n            'name': 'Huobi CNY',\n            'hostname': 'be.huobi.com',\n            'has': {\n                'CORS': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766569-15aa7b9a-5edd-11e7-9e7f-44791f4ee49c.jpg',\n                'api': 'https://be.huobi.com',\n                'www': 'https://www.huobi.com',\n                'doc': 'https://github.com/huobiapi/API_Docs/wiki/REST_api_reference',\n            },\n        });\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class huobipro extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'huobipro',\n            'name': 'Huobi Pro',\n            'countries': 'CN',\n            'rateLimit': 2000,\n            'userAgent': this.userAgents['chrome39'],\n            'version': 'v1',\n            'accounts': undefined,\n            'accountsById': undefined,\n            'hostname': 'api.huobi.pro',\n            'has': {\n                'CORS': false,\n                'fetchOHCLV': true,\n                'fetchOrders': true,\n                'fetchOpenOrders': true,\n            },\n            'timeframes': {\n                '1m': '1min',\n                '5m': '5min',\n                '15m': '15min',\n                '30m': '30min',\n                '1h': '60min',\n                '1d': '1day',\n                '1w': '1week',\n                '1M': '1mon',\n                '1y': '1year',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766569-15aa7b9a-5edd-11e7-9e7f-44791f4ee49c.jpg',\n                'api': 'https://api.huobi.pro',\n                'www': 'https://www.huobi.pro',\n                'doc': 'https://github.com/huobiapi/API_Docs/wiki/REST_api_reference',\n            },\n            'api': {\n                'market': {\n                    'get': [\n                        'history/kline', // 获取K线数据\n                        'detail/merged', // 获取聚合行情(Ticker)\n                        'depth', // 获取 Market Depth 数据\n                        'trade', // 获取 Trade Detail 数据\n                        'history/trade', // 批量获取最近的交易记录\n                        'detail', // 获取 Market Detail 24小时成交量数据\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'common/symbols', // 查询系统支持的所有交易对\n                        'common/currencys', // 查询系统支持的所有币种\n                        'common/timestamp', // 查询系统当前时间\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'account/accounts', // 查询当前用户的所有账户(即account-id)\n                        'account/accounts/{id}/balance', // 查询指定账户的余额\n                        'order/orders/{id}', // 查询某个订单详情\n                        'order/orders/{id}/matchresults', // 查询某个订单的成交明细\n                        'order/orders', // 查询当前委托、历史委托\n                        'order/matchresults', // 查询当前成交、历史成交\n                        'dw/withdraw-virtual/addresses', // 查询虚拟币提现地址\n                    ],\n                    'post': [\n                        'order/orders/place', // 创建并执行一个新订单 (一步下单， 推荐使用)\n                        'order/orders', // 创建一个新的订单请求 （仅创建订单，不执行下单）\n                        'order/orders/{id}/place', // 执行一个订单 （仅执行已创建的订单）\n                        'order/orders/{id}/submitcancel', // 申请撤销一个订单请求\n                        'order/orders/batchcancel', // 批量撤销订单\n                        'dw/balance/transfer', // 资产划转\n                        'dw/withdraw-virtual/create', // 申请提现虚拟币\n                        'dw/withdraw-virtual/{id}/place', // 确认申请虚拟币提现\n                        'dw/withdraw-virtual/{id}/cancel', // 申请取消提现虚拟币\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicGetCommonSymbols ();\n        let markets = response['data'];\n        let numMarkets = markets.length;\n        if (numMarkets < 1)\n            throw new ExchangeError (this.id + ' publicGetCommonSymbols returned empty response: ' + this.json (response));\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let baseId = market['base-currency'];\n            let quoteId = market['quote-currency'];\n            let base = baseId.toUpperCase ();\n            let quote = quoteId.toUpperCase ();\n            let id = baseId + quoteId;\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            let precision = {\n                'amount': market['amount-precision'],\n                'price': market['price-precision'],\n            };\n            let lot = Math.pow (10, -precision['amount']);\n            let maker = (base == 'OMG') ? 0 : 0.2 / 100;\n            let taker = (base == 'OMG') ? 0 : 0.2 / 100;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'lot': lot,\n                'precision': precision,\n                'taker': taker,\n                'maker': maker,\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': Math.pow (10, precision['amount']),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision['price']),\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': 0,\n                        'max': undefined,\n                    },\n                },\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let last = undefined;\n        if ('last' in ticker)\n            last = ticker['last'];\n        let timestamp = this.milliseconds ();\n        if ('ts' in ticker)\n            timestamp = ticker['ts'];\n        let bid = undefined;\n        let ask = undefined;\n        if ('bid' in ticker) {\n            if (ticker['bid'])\n                bid = this.safeFloat (ticker['bid'], 0);\n        }\n        if ('ask' in ticker) {\n            if (ticker['ask'])\n                ask = this.safeFloat (ticker['ask'], 0);\n        }\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': ticker['high'],\n            'low': ticker['low'],\n            'bid': bid,\n            'ask': ask,\n            'vwap': undefined,\n            'open': ticker['open'],\n            'close': ticker['close'],\n            'first': undefined,\n            'last': last,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['amount']),\n            'quoteVolume': ticker['vol'],\n            'info': ticker,\n        };\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.marketGetDepth (this.extend ({\n            'symbol': market['id'],\n            'type': 'step0',\n        }, params));\n        if ('tick' in response) {\n            if (!response['tick']) {\n                throw new ExchangeError (this.id + ' fetchOrderBook() returned empty response: ' + this.json (response));\n            }\n            return this.parseOrderBook (response['tick'], response['tick']['ts']);\n        }\n        throw new ExchangeError (this.id + ' fetchOrderBook() returned unrecognized response: ' + this.json (response));\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.marketGetDetailMerged (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTicker (response['tick'], market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['ts'];\n        return {\n            'info': trade,\n            'id': trade['id'].toString (),\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['direction'],\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    parseTradesData (data, market, since = undefined, limit = undefined) {\n        let result = [];\n        for (let i = 0; i < data.length; i++) {\n            let trades = this.parseTrades (data[i]['data'], market, since, limit);\n            for (let k = 0; k < trades.length; k++) {\n                result.push (trades[k]);\n            }\n        }\n        return result;\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.marketGetHistoryTrade (this.extend ({\n            'symbol': market['id'],\n            'size': 2000,\n        }, params));\n        return this.parseTradesData (response['data'], market, since, limit);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv['id'] * 1000,\n            ohlcv['open'],\n            ohlcv['high'],\n            ohlcv['low'],\n            ohlcv['close'],\n            ohlcv['vol'],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.marketGetHistoryKline (this.extend ({\n            'symbol': market['id'],\n            'period': this.timeframes[timeframe],\n            'size': 2000, // max = 2000\n        }, params));\n        return this.parseOHLCVs (response['data'], market, timeframe, since, limit);\n    }\n\n    async loadAccounts (reload = false) {\n        if (reload) {\n            this.accounts = await this.fetchAccounts ();\n        } else {\n            if (this.accounts) {\n                return this.accounts;\n            } else {\n                this.accounts = await this.fetchAccounts ();\n                this.accountsById = this.indexBy (this.accounts, 'id');\n            }\n        }\n        return this.accounts;\n    }\n\n    async fetchAccounts () {\n        await this.loadMarkets ();\n        let response = await this.privateGetAccountAccounts ();\n        return response['data'];\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        await this.loadAccounts ();\n        let response = await this.privateGetAccountAccountsIdBalance (this.extend ({\n            'id': this.accounts[0]['id'],\n        }, params));\n        let balances = response['data']['list'];\n        let result = { 'info': response };\n        for (let i = 0; i < balances.length; i++) {\n            let balance = balances[i];\n            let uppercase = balance['currency'].toUpperCase ();\n            let currency = this.commonCurrencyCode (uppercase);\n            let account = undefined;\n            if (currency in result)\n                account = result[currency];\n            else\n                account = this.account ();\n            if (balance['type'] == 'trade')\n                account['free'] = parseFloat (balance['balance']);\n            if (balance['type'] == 'frozen')\n                account['used'] = parseFloat (balance['balance']);\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrders() requires a symbol parameter');\n        this.load_markets ();\n        let market = this.market (symbol);\n        let status = undefined;\n        if ('type' in params) {\n            status = params['type'];\n        } else if ('status' in params) {\n            status = params['status'];\n        } else {\n            throw new ExchangeError (this.id + ' fetchOrders() requires type param or status param for spot market ' + symbol + '(0 or \"open\" for unfilled or partial filled orders, 1 or \"closed\" for filled orders)');\n        }\n        if ((status == 0) || (status == 'open')) {\n            status = 'submitted,partial-filled';\n        } else if ((status == 1) || (status == 'closed')) {\n            status = 'filled,partial-canceled';\n        } else {\n            throw new ExchangeError (this.id + ' fetchOrders() wrong type param or status param for spot market ' + symbol + '(0 or \"open\" for unfilled or partial filled orders, 1 or \"closed\" for filled orders)');\n        }\n        let response = await this.privateGetOrderOrders (this.extend ({\n            'symbol': market['id'],\n            'states': status,\n        }));\n        return this.parseOrders (response['data'], market, since, limit);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let open = 0; // 0 for unfilled orders, 1 for filled orders\n        return this.fetchOrders (symbol, undefined, undefined, this.extend ({\n            'status': open,\n        }, params));\n    }\n\n    parseOrderStatus (status) {\n        if (status == 'partial-filled') {\n            return 'open';\n        } else if (status == 'filled') {\n            return 'closed';\n        } else if (status == 'canceled') {\n            return 'canceled';\n        } else if (status == 'submitted') {\n            return 'open';\n        }\n        return status;\n    }\n\n    parseOrder (order, market = undefined) {\n        let side = undefined;\n        let type = undefined;\n        let status = undefined;\n        if ('type' in order) {\n            let orderType = order['type'].split ('-');\n            side = orderType[0];\n            type = orderType[1];\n            status = this.parseOrderStatus (order['state']);\n        }\n        let symbol = undefined;\n        if (!market) {\n            if ('symbol' in order) {\n                if (order['symbol'] in this.markets_by_id) {\n                    let marketId = order['symbol'];\n                    market = this.markets_by_id[marketId];\n                }\n            }\n        }\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = order['created-at'];\n        let amount = parseFloat (order['amount']);\n        let filled = parseFloat (order['field-amount']);\n        let remaining = amount - filled;\n        let price = parseFloat (order['price']);\n        let cost = parseFloat (order['field-cash-amount']);\n        let average = 0;\n        if (filled)\n            average = parseFloat (cost / filled);\n        let result = {\n            'info': order,\n            'id': order['id'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': price,\n            'average': average,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': undefined,\n        };\n        return result;\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        await this.loadAccounts ();\n        let market = this.market (symbol);\n        let order = {\n            'account-id': this.accounts[0]['id'],\n            'amount': this.amountToPrecision (symbol, amount),\n            'symbol': market['id'],\n            'type': side + '-' + type,\n        };\n        if (type == 'limit')\n            order['price'] = this.priceToPrecision (symbol, price);\n        let response = await this.privatePostOrderOrdersPlace (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['data'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostOrderOrdersIdSubmitcancel ({ 'id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = '/';\n        if (api == 'market')\n            url += api;\n        else\n            url += this.version;\n        url += '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            let timestamp = this.YmdHMS (this.milliseconds (), 'T');\n            let request = this.keysort (this.extend ({\n                'SignatureMethod': 'HmacSHA256',\n                'SignatureVersion': '2',\n                'AccessKeyId': this.apiKey,\n                'Timestamp': timestamp,\n            }, query));\n            let auth = this.urlencode (request);\n            let payload = [ method, this.hostname, url, auth ].join (\"\\n\");\n            let signature = this.hmac (this.encode (payload), this.encode (this.secret), 'sha256', 'base64');\n            auth += '&' + this.urlencode ({ 'Signature': signature });\n            url += '?' + auth;\n            if (method == 'POST') {\n                body = this.json (query);\n                headers = {\n                    'Content-Type': 'application/json',\n                };\n            }\n        } else {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        }\n        url = this.urls['api'] + url;\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('status' in response)\n            if (response['status'] == 'error')\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class independentreserve extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'independentreserve',\n            'name': 'Independent Reserve',\n            'countries': [ 'AU', 'NZ' ], // Australia, New Zealand\n            'rateLimit': 1000,\n            'has': {\n                'CORS': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30521662-cf3f477c-9bcb-11e7-89bc-d1ac85012eda.jpg',\n                'api': {\n                    'public': 'https://api.independentreserve.com/Public',\n                    'private': 'https://api.independentreserve.com/Private',\n                },\n                'www': 'https://www.independentreserve.com',\n                'doc': 'https://www.independentreserve.com/API',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'GetValidPrimaryCurrencyCodes',\n                        'GetValidSecondaryCurrencyCodes',\n                        'GetValidLimitOrderTypes',\n                        'GetValidMarketOrderTypes',\n                        'GetValidOrderTypes',\n                        'GetValidTransactionTypes',\n                        'GetMarketSummary',\n                        'GetOrderBook',\n                        'GetTradeHistorySummary',\n                        'GetRecentTrades',\n                        'GetFxRates',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'PlaceLimitOrder',\n                        'PlaceMarketOrder',\n                        'CancelOrder',\n                        'GetOpenOrders',\n                        'GetClosedOrders',\n                        'GetClosedFilledOrders',\n                        'GetOrderDetails',\n                        'GetAccounts',\n                        'GetTransactions',\n                        'GetDigitalCurrencyDepositAddress',\n                        'GetDigitalCurrencyDepositAddresses',\n                        'SynchDigitalCurrencyDepositAddressWithBlockchain',\n                        'WithdrawDigitalCurrency',\n                        'RequestFiatWithdrawal',\n                        'GetTrades',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let baseCurrencies = await this.publicGetValidPrimaryCurrencyCodes ();\n        let quoteCurrencies = await this.publicGetValidSecondaryCurrencyCodes ();\n        let result = [];\n        for (let i = 0; i < baseCurrencies.length; i++) {\n            let baseId = baseCurrencies[i];\n            let baseIdUppercase = baseId.toUpperCase ();\n            let base = this.commonCurrencyCode (baseIdUppercase);\n            for (let j = 0; j < quoteCurrencies.length; j++) {\n                let quoteId = quoteCurrencies[j];\n                let quoteIdUppercase = quoteId.toUpperCase ();\n                let quote = this.commonCurrencyCode (quoteIdUppercase);\n                let id = baseId + '/' + quoteId;\n                let symbol = base + '/' + quote;\n                let taker = 0.5 / 100;\n                let maker = 0.5 / 100;\n                result.push ({\n                    'id': id,\n                    'symbol': symbol,\n                    'base': base,\n                    'quote': quote,\n                    'baseId': baseId,\n                    'quoteId': quoteId,\n                    'taker': taker,\n                    'maker': maker,\n                    'info': id,\n                });\n            }\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privatePostGetAccounts ();\n        let result = { 'info': balances };\n        for (let i = 0; i < balances.length; i++) {\n            let balance = balances[i];\n            let currencyCode = balance['CurrencyCode'];\n            let uppercase = currencyCode.toUpperCase ();\n            let currency = this.commonCurrencyCode (uppercase);\n            let account = this.account ();\n            account['free'] = balance['AvailableBalance'];\n            account['total'] = balance['TotalBalance'];\n            account['used'] = account['total'] - account['free'];\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetOrderBook (this.extend ({\n            'primaryCurrencyCode': market['baseId'],\n            'secondaryCurrencyCode': market['quoteId'],\n        }, params));\n        let timestamp = this.parse8601 (response['CreatedTimestampUtc']);\n        return this.parseOrderBook (response, timestamp, 'BuyOrders', 'SellOrders', 'Price', 'Volume');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.parse8601 (ticker['CreatedTimestampUtc']);\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': ticker['DayHighestPrice'],\n            'low': ticker['DayLowestPrice'],\n            'bid': ticker['CurrentHighestBidPrice'],\n            'ask': ticker['CurrentLowestOfferPrice'],\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': ticker['LastPrice'],\n            'change': undefined,\n            'percentage': undefined,\n            'average': ticker['DayAvgPrice'],\n            'baseVolume': ticker['DayVolumeXbtInSecondaryCurrrency'],\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetMarketSummary (this.extend ({\n            'primaryCurrencyCode': market['baseId'],\n            'secondaryCurrencyCode': market['quoteId'],\n        }, params));\n        return this.parseTicker (response, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = this.parse8601 (trade['TradeTimestampUtc']);\n        return {\n            'id': undefined,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': undefined,\n            'type': undefined,\n            'side': undefined,\n            'price': trade['SecondaryCurrencyTradePrice'],\n            'amount': trade['PrimaryCurrencyAmount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetRecentTrades (this.extend ({\n            'primaryCurrencyCode': market['baseId'],\n            'secondaryCurrencyCode': market['quoteId'],\n            'numberOfRecentTradesToRetrieve': 50, // max = 50\n        }, params));\n        return this.parseTrades (response['Trades'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let capitalizedOrderType = this.capitalize (type);\n        let method = 'privatePostPlace' + capitalizedOrderType + 'Order';\n        let orderType = capitalizedOrderType;\n        orderType += (side == 'sell') ?  'Offer' : 'Bid';\n        let order = this.ordered ({\n            'primaryCurrencyCode': market['baseId'],\n            'secondaryCurrencyCode': market['quoteId'],\n            'orderType': orderType,\n        });\n        if (type == 'limit')\n            order['price'] = price;\n        order['volume'] = amount;\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['OrderGuid'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelOrder ({ 'orderGuid': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api] + '/' + path;\n        if (api == 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let auth = [\n                url,\n                'apiKey=' + this.apiKey,\n                'nonce=' + nonce.toString (),\n            ];\n            let keysorted = this.keysort (params);\n            let keys = Object.keys (keysorted);\n            for (let i = 0; i < keys.length; i++) {\n                let key = keys[i];\n                auth.push (key + '=' + params[key]);\n            }\n            let message = auth.join (',');\n            let signature = this.hmac (this.encode (message), this.encode (this.secret));\n            let query = this.keysort (this.extend ({\n                'apiKey': this.apiKey,\n                'nonce': nonce,\n                'signature': signature,\n            }, params));\n            body = this.json (query);\n            headers = { 'Content-Type': 'application/json' };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        // todo error handling\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class itbit extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'itbit',\n            'name': 'itBit',\n            'countries': 'US',\n            'rateLimit': 2000,\n            'version': 'v1',\n            'has': {\n                'CORS': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27822159-66153620-60ad-11e7-89e7-005f6d7f3de0.jpg',\n                'api': 'https://api.itbit.com',\n                'www': 'https://www.itbit.com',\n                'doc': [\n                    'https://api.itbit.com/docs',\n                    'https://www.itbit.com/api',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'markets/{symbol}/ticker',\n                        'markets/{symbol}/order_book',\n                        'markets/{symbol}/trades',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'wallets',\n                        'wallets/{walletId}',\n                        'wallets/{walletId}/balances/{currencyCode}',\n                        'wallets/{walletId}/funding_history',\n                        'wallets/{walletId}/trades',\n                        'wallets/{walletId}/orders/{id}',\n                    ],\n                    'post': [\n                        'wallet_transfers',\n                        'wallets',\n                        'wallets/{walletId}/cryptocurrency_deposits',\n                        'wallets/{walletId}/cryptocurrency_withdrawals',\n                        'wallets/{walletId}/orders',\n                        'wire_withdrawal',\n                    ],\n                    'delete': [\n                        'wallets/{walletId}/orders/{id}',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/USD': { 'id': 'XBTUSD', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD' },\n                'BTC/SGD': { 'id': 'XBTSGD', 'symbol': 'BTC/SGD', 'base': 'BTC', 'quote': 'SGD' },\n                'BTC/EUR': { 'id': 'XBTEUR', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR' },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0,\n                    'taker': 0.2 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetMarketsSymbolOrderBook (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetMarketsSymbolTicker (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        let serverTimeUTC = ('serverTimeUTC' in ticker);\n        if (!serverTimeUTC)\n            throw new ExchangeError (this.id + ' fetchTicker returned a bad response: ' + this.json (ticker));\n        let timestamp = this.parse8601 (ticker['serverTimeUTC']);\n        let vwap = parseFloat (ticker['vwap24h']);\n        let baseVolume = parseFloat (ticker['volume24h']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high24h']),\n            'low': parseFloat (ticker['low24h']),\n            'bid': this.safeFloat (ticker, 'bid'),\n            'ask': this.safeFloat (ticker, 'ask'),\n            'vwap': vwap,\n            'open': parseFloat (ticker['openToday']),\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['lastPrice']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = this.parse8601 (trade['timestamp']);\n        let id = trade['matchNumber'].toString ();\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': id,\n            'order': id,\n            'type': undefined,\n            'side': undefined,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetMarketsSymbolTrades (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTrades (response['recentTrades'], market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privateGetBalances ();\n        let balances = response['balances'];\n        let result = { 'info': response };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let account = {\n                'free': parseFloat (balance['availableBalance']),\n                'used': 0.0,\n                'total': parseFloat (balance['totalBalance']),\n            };\n            account['used'] = account['total'] - account['free'];\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    fetchWallets () {\n        return this.privateGetWallets ();\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type == 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let walletIdInParams = ('walletId' in params);\n        if (!walletIdInParams)\n            throw new ExchangeError (this.id + ' createOrder requires a walletId parameter');\n        amount = amount.toString ();\n        price = price.toString ();\n        let market = this.market (symbol);\n        let order = {\n            'side': side,\n            'type': type,\n            'currency': market['base'],\n            'amount': amount,\n            'display': amount,\n            'price': price,\n            'instrument': market['id'],\n        };\n        let response = await this.privatePostTradeAdd (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        let walletIdInParams = ('walletId' in params);\n        if (!walletIdInParams)\n            throw new ExchangeError (this.id + ' cancelOrder requires a walletId parameter');\n        return await this.privateDeleteWalletsWalletIdOrdersId (this.extend ({\n            'id': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            if (Object.keys (query).length)\n                body = this.json (query);\n            else\n                body = '';\n            let nonce = this.nonce ().toString ();\n            let timestamp = nonce;\n            let auth = [ method, url, body, nonce, timestamp ];\n            let message = nonce + this.json (auth);\n            let hash = this.hash (this.encode (message), 'sha256', 'binary');\n            let binhash = this.binaryConcat (url, hash);\n            let signature = this.hmac (binhash, this.encode (this.secret), 'sha512', 'base64');\n            headers = {\n                'Authorization': this.apiKey + ':' + signature,\n                'Content-Type': 'application/json',\n                'X-Auth-Timestamp': timestamp,\n                'X-Auth-Nonce': nonce,\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('code' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst btcbox = require ('./btcbox.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class jubi extends btcbox {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'jubi',\n            'name': 'jubi.com',\n            'countries': 'CN',\n            'rateLimit': 1500,\n            'version': 'v1',\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766581-9d397d9a-5edd-11e7-8fb9-5d8236c0e692.jpg',\n                'api': 'https://www.jubi.com/api',\n                'www': 'https://www.jubi.com',\n                'doc': 'https://www.jubi.com/help/api.html',\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetAllticker ();\n        let keys = Object.keys (markets);\n        let result = [];\n        for (let p = 0; p < keys.length; p++) {\n            let id = keys[p];\n            let base = id.toUpperCase ();\n            let quote = 'CNY'; // todo\n            let symbol = base + '/' + quote;\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': id,\n            });\n        }\n        return result;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeNotAvailable, ExchangeError, OrderNotFound, DDoSProtection, InvalidNonce, InsufficientFunds, CancelPending, InvalidOrder } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class kraken extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'kraken',\n            'name': 'Kraken',\n            'countries': 'US',\n            'version': '0',\n            'rateLimit': 3000,\n            'has': {\n                'CORS': false,\n                'fetchCurrencies': true,\n                'fetchTickers': true,\n                'fetchOHLCV': true,\n                'fetchOrder': true,\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': true,\n                'fetchMyTrades': true,\n                'withdraw': true,\n            },\n            'marketsByAltname': {},\n            'timeframes': {\n                '1m': '1',\n                '5m': '5',\n                '15m': '15',\n                '30m': '30',\n                '1h': '60',\n                '4h': '240',\n                '1d': '1440',\n                '1w': '10080',\n                '2w': '21600',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766599-22709304-5ede-11e7-9de1-9f33732e1509.jpg',\n                'api': 'https://api.kraken.com',\n                'www': 'https://www.kraken.com',\n                'doc': [\n                    'https://www.kraken.com/en-us/help/api',\n                    'https://github.com/nothingisdead/npm-kraken-api',\n                ],\n                'fees': [\n                    'https://www.kraken.com/en-us/help/fees',\n                    'https://support.kraken.com/hc/en-us/articles/201396777-What-are-the-deposit-fees-',\n                    'https://support.kraken.com/hc/en-us/articles/201893608-What-are-the-withdrawal-fees-',\n                ],\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': true,\n                    'percentage': true,\n                    'taker': 0.26 / 100,\n                    'maker': 0.16 / 100,\n                    'tiers': {\n                        'taker': [\n                            [0, 0.26 / 100],\n                            [50000, 0.24 / 100],\n                            [100000, 0.22 / 100],\n                            [250000, 0.2 / 100],\n                            [500000, 0.18 / 100],\n                            [1000000, 0.16 / 100],\n                            [2500000, 0.14 / 100],\n                            [5000000, 0.12 / 100],\n                            [10000000, 0.1 / 100],\n                        ],\n                        'maker': [\n                            [0, 0.16 / 100],\n                            [50000, 0.14 / 100],\n                            [100000, 0.12 / 100],\n                            [250000, 0.10 / 100],\n                            [500000, 0.8 / 100],\n                            [1000000, 0.6 / 100],\n                            [2500000, 0.4 / 100],\n                            [5000000, 0.2 / 100],\n                            [10000000, 0.0 / 100],\n                        ],\n                    },\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.001,\n                        'ETH': 0.005,\n                        'XRP': 0.02,\n                        'XLM': 0.00002,\n                        'LTC': 0.02,\n                        'DOGE': 2,\n                        'ZEC': 0.00010,\n                        'ICN': 0.02,\n                        'REP': 0.01,\n                        'ETC': 0.005,\n                        'MLN': 0.003,\n                        'XMR': 0.05,\n                        'DASH': 0.005,\n                        'GNO': 0.01,\n                        'EOS': 0.5,\n                        'BCH': 0.001,\n                        'USD': 5, // if domestic wire\n                        'EUR': 5, // if domestic wire\n                        'CAD': 10, // CAD EFT Withdrawal\n                        'JPY': 300, // if domestic wire\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                        'ETH': 0,\n                        'XRP': 0,\n                        'XLM': 0,\n                        'LTC': 0,\n                        'DOGE': 0,\n                        'ZEC': 0,\n                        'ICN': 0,\n                        'REP': 0,\n                        'ETC': 0,\n                        'MLN': 0,\n                        'XMR': 0,\n                        'DASH': 0,\n                        'GNO': 0,\n                        'EOS': 0,\n                        'BCH': 0,\n                        'USD': 5, // if domestic wire\n                        'EUR': 0, // free deposit if EUR SEPA Deposit\n                        'CAD': 5, // if domestic wire\n                        'JPY': 0, // Domestic Deposit (Free, ¥5,000 deposit minimum)\n                    },\n                },\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'Assets',\n                        'AssetPairs',\n                        'Depth',\n                        'OHLC',\n                        'Spread',\n                        'Ticker',\n                        'Time',\n                        'Trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'AddOrder',\n                        'Balance',\n                        'CancelOrder',\n                        'ClosedOrders',\n                        'DepositAddresses',\n                        'DepositMethods',\n                        'DepositStatus',\n                        'Ledgers',\n                        'OpenOrders',\n                        'OpenPositions',\n                        'QueryLedgers',\n                        'QueryOrders',\n                        'QueryTrades',\n                        'TradeBalance',\n                        'TradesHistory',\n                        'TradeVolume',\n                        'Withdraw',\n                        'WithdrawCancel',\n                        'WithdrawInfo',\n                        'WithdrawStatus',\n                    ],\n                },\n            },\n        });\n    }\n\n    costToPrecision (symbol, cost) {\n        return this.truncate (parseFloat (cost), this.markets[symbol]['precision']['price']);\n    }\n\n    feeToPrecision (symbol, fee) {\n        return this.truncate (parseFloat (fee), this.markets[symbol]['precision']['amount']);\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (body.indexOf ('Invalid order') >= 0)\n            throw new InvalidOrder (this.id + ' ' + body);\n        if (body.indexOf ('Invalid nonce') >= 0)\n            throw new InvalidNonce (this.id + ' ' + body);\n        if (body.indexOf ('Insufficient funds') >= 0)\n            throw new InsufficientFunds (this.id + ' ' + body);\n        if (body.indexOf ('Cancel pending') >= 0)\n            throw new CancelPending (this.id + ' ' + body);\n        if (body.indexOf ('Invalid arguments:volume') >= 0)\n            throw new InvalidOrder (this.id + ' ' + body);\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetAssetPairs ();\n        let keys = Object.keys (markets['result']);\n        let result = [];\n        for (let i = 0; i < keys.length; i++) {\n            let id = keys[i];\n            let market = markets['result'][id];\n            let base = market['base'];\n            let quote = market['quote'];\n            if ((base[0] === 'X') || (base[0] === 'Z'))\n                base = base.slice (1);\n            if ((quote[0] === 'X') || (quote[0] === 'Z'))\n                quote = quote.slice (1);\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let darkpool = id.indexOf ('.d') >= 0;\n            let symbol = darkpool ? market['altname'] : (base + '/' + quote);\n            let maker = undefined;\n            if ('fees_maker' in market) {\n                maker = parseFloat (market['fees_maker'][0][1]) / 100;\n            }\n            let precision = {\n                'amount': market['lot_decimals'],\n                'price': market['pair_decimals'],\n            };\n            let lot = Math.pow (10, -precision['amount']);\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'darkpool': darkpool,\n                'info': market,\n                'altname': market['altname'],\n                'maker': maker,\n                'taker': parseFloat (market['fees'][0][1]) / 100,\n                'lot': lot,\n                'active': true,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': Math.pow (10, precision['amount']),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision['price']),\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': 0,\n                        'max': undefined,\n                    },\n                },\n            });\n        }\n        result = this.appendInactiveMarkets (result);\n        this.marketsByAltname = this.indexBy (result, 'altname');\n        return result;\n    }\n\n    appendInactiveMarkets (result = []) {\n        let precision = { 'amount': 8, 'price': 8 };\n        let costLimits = { 'min': 0, 'max': undefined };\n        let priceLimits = { 'min': Math.pow (10, -precision['price']), 'max': undefined };\n        let amountLimits = { 'min': Math.pow (10, -precision['amount']), 'max': Math.pow (10, precision['amount']) };\n        let limits = { 'amount': amountLimits, 'price': priceLimits, 'cost': costLimits };\n        let defaults = {\n            'darkpool': false,\n            'info': undefined,\n            'maker': undefined,\n            'taker': undefined,\n            'lot': amountLimits['min'],\n            'active': false,\n            'precision': precision,\n            'limits': limits,\n        };\n        let markets = [\n            { 'id': 'XXLMZEUR', 'symbol': 'XLM/EUR', 'base': 'XLM', 'quote': 'EUR', 'altname': 'XLMEUR' },\n        ];\n        for (let i = 0; i < markets.length; i++) {\n            result.push (this.extend (defaults, markets[i]));\n        }\n        return result;\n    }\n\n    async fetchCurrencies (params = {}) {\n        let response = await this.publicGetAssets (params);\n        let currencies = response['result'];\n        let ids = Object.keys (currencies);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let currency = currencies[id];\n            // todo: will need to rethink the fees\n            // to add support for multiple withdrawal/deposit methods and\n            // differentiated fees for each particular method\n            let code = this.commonCurrencyCode (currency['altname']);\n            let precision = currency['decimals'];\n            result[code] = {\n                'id': id,\n                'code': code,\n                'info': currency,\n                'name': code,\n                'active': true,\n                'status': 'ok',\n                'fee': undefined,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': undefined,\n                        'max': Math.pow (10, precision),\n                    },\n                },\n            };\n        }\n        return result;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let darkpool = symbol.indexOf ('.d') >= 0;\n        if (darkpool)\n            throw new ExchangeError (this.id + ' does not provide an order book for darkpool symbol ' + symbol);\n        let market = this.market (symbol);\n        let response = await this.publicGetDepth (this.extend ({\n            'pair': market['id'],\n            // 'count': 100,\n        }, params));\n        let orderbook = response['result'][market['id']];\n        return this.parseOrderBook (orderbook);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let baseVolume = parseFloat (ticker['v'][1]);\n        let vwap = parseFloat (ticker['p'][1]);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['h'][1]),\n            'low': parseFloat (ticker['l'][1]),\n            'bid': parseFloat (ticker['b'][0]),\n            'ask': parseFloat (ticker['a'][0]),\n            'vwap': vwap,\n            'open': parseFloat (ticker['o']),\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['c'][0]),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let pairs = [];\n        for (let s = 0; s < this.symbols.length; s++) {\n            let symbol = this.symbols[s];\n            let market = this.markets[symbol];\n            if (market['active'])\n                if (!market['darkpool'])\n                    pairs.push (market['id']);\n        }\n        let filter = pairs.join (',');\n        let response = await this.publicGetTicker (this.extend ({\n            'pair': filter,\n        }, params));\n        let tickers = response['result'];\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let darkpool = symbol.indexOf ('.d') >= 0;\n        if (darkpool)\n            throw new ExchangeError (this.id + ' does not provide a ticker for darkpool symbol ' + symbol);\n        let market = this.market (symbol);\n        let response = await this.publicGetTicker (this.extend ({\n            'pair': market['id'],\n        }, params));\n        let ticker = response['result'][market['id']];\n        return this.parseTicker (ticker, market);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv[0] * 1000,\n            parseFloat (ohlcv[1]),\n            parseFloat (ohlcv[2]),\n            parseFloat (ohlcv[3]),\n            parseFloat (ohlcv[4]),\n            parseFloat (ohlcv[6]),\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'pair': market['id'],\n            'interval': this.timeframes[timeframe],\n        };\n        if (since)\n            request['since'] = parseInt (since / 1000);\n        let response = await this.publicGetOHLC (this.extend (request, params));\n        let ohlcvs = response['result'][market['id']];\n        return this.parseOHLCVs (ohlcvs, market, timeframe, since, limit);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = undefined;\n        let side = undefined;\n        let type = undefined;\n        let price = undefined;\n        let amount = undefined;\n        let id = undefined;\n        let order = undefined;\n        let fee = undefined;\n        if (!market)\n            market = this.findMarketByAltnameOrId (trade['pair']);\n        if ('ordertxid' in trade) {\n            order = trade['ordertxid'];\n            id = trade['id'];\n            timestamp = parseInt (trade['time'] * 1000);\n            side = trade['type'];\n            type = trade['ordertype'];\n            price = parseFloat (trade['price']);\n            amount = parseFloat (trade['vol']);\n            if ('fee' in trade) {\n                let currency = undefined;\n                if (market)\n                    currency = market['quote'];\n                fee = {\n                    'cost': parseFloat (trade['fee']),\n                    'currency': currency,\n                };\n            }\n        } else {\n            timestamp = parseInt (trade[2] * 1000);\n            side = (trade[3] === 's') ? 'sell' : 'buy';\n            type = (trade[4] === 'l') ? 'limit' : 'market';\n            price = parseFloat (trade[0]);\n            amount = parseFloat (trade[1]);\n        }\n        let symbol = (market) ? market['symbol'] : undefined;\n        return {\n            'id': id,\n            'order': order,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': price,\n            'amount': amount,\n            'fee': fee,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let id = market['id'];\n        let response = await this.publicGetTrades (this.extend ({\n            'pair': id,\n        }, params));\n        let trades = response['result'][id];\n        return this.parseTrades (trades, market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostBalance ();\n        let balances = response['result'];\n        let result = { 'info': balances };\n        let currencies = Object.keys (balances);\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let code = currency;\n            // X-ISO4217-A3 standard currency codes\n            if (code[0] === 'X') {\n                code = code.slice (1);\n            } else if (code[0] === 'Z') {\n                code = code.slice (1);\n            }\n            code = this.commonCurrencyCode (code);\n            let balance = parseFloat (balances[currency]);\n            let account = {\n                'free': balance,\n                'used': 0.0,\n                'total': balance,\n            };\n            result[code] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let order = {\n            'pair': market['id'],\n            'type': side,\n            'ordertype': type,\n            'volume': this.amountToPrecision (symbol, amount),\n        };\n        if (type === 'limit')\n            order['price'] = this.priceToPrecision (symbol, price);\n        let response = await this.privatePostAddOrder (this.extend (order, params));\n        let length = response['result']['txid'].length;\n        let id = (length > 1) ? response['result']['txid'] : response['result']['txid'][0];\n        return {\n            'info': response,\n            'id': id,\n        };\n    }\n\n    findMarketByAltnameOrId (id) {\n        let result = undefined;\n        if (id in this.marketsByAltname) {\n            result = this.marketsByAltname[id];\n        } else if (id in this.markets_by_id) {\n            result = this.markets_by_id[id];\n        }\n        return result;\n    }\n\n    parseOrder (order, market = undefined) {\n        let description = order['descr'];\n        let side = description['type'];\n        let type = description['ordertype'];\n        let symbol = undefined;\n        if (!market)\n            market = this.findMarketByAltnameOrId (description['pair']);\n        let timestamp = parseInt (order['opentm'] * 1000);\n        let amount = parseFloat (order['vol']);\n        let filled = parseFloat (order['vol_exec']);\n        let remaining = amount - filled;\n        let fee = undefined;\n        let cost = this.safeFloat (order, 'cost');\n        let price = this.safeFloat (description, 'price');\n        if (!price)\n            price = this.safeFloat (order, 'price');\n        if (market) {\n            symbol = market['symbol'];\n            if ('fee' in order) {\n                let flags = order['oflags'];\n                let feeCost = this.safeFloat (order, 'fee');\n                fee = {\n                    'cost': feeCost,\n                    'rate': undefined,\n                };\n                if (flags.indexOf ('fciq') >= 0) {\n                    fee['currency'] = market['quote'];\n                } else if (flags.indexOf ('fcib') >= 0) {\n                    fee['currency'] = market['base'];\n                }\n            }\n        }\n        return {\n            'id': order['id'],\n            'info': order,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': order['status'],\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'fee': fee,\n            // 'trades': this.parseTrades (order['trades'], market),\n        };\n    }\n\n    parseOrders (orders, market = undefined, since = undefined, limit = undefined) {\n        let result = [];\n        let ids = Object.keys (orders);\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let order = this.extend ({ 'id': id }, orders[id]);\n            result.push (this.parseOrder (order, market));\n        }\n        return this.filterBySinceLimit (result, since, limit);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostQueryOrders (this.extend ({\n            'trades': true, // whether or not to include trades in output (optional, default false)\n            'txid': id, // comma delimited list of transaction ids to query info about (20 maximum)\n            // 'userref': 'optional', // restrict results to given user reference id (optional)\n        }, params));\n        let orders = response['result'];\n        let order = this.parseOrder (this.extend ({ 'id': id }, orders[id]));\n        return this.extend ({ 'info': response }, order);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            // 'type': 'all', // any position, closed position, closing position, no position\n            // 'trades': false, // whether or not to include trades related to position in output\n            // 'start': 1234567890, // starting unix timestamp or trade tx id of results (exclusive)\n            // 'end': 1234567890, // ending unix timestamp or trade tx id of results (inclusive)\n            // 'ofs' = result offset\n        };\n        if (since)\n            request['start'] = parseInt (since / 1000);\n        let response = await this.privatePostTradesHistory (this.extend (request, params));\n        let trades = response['result']['trades'];\n        let ids = Object.keys (trades);\n        for (let i = 0; i < ids.length; i++) {\n            trades[ids[i]]['id'] = ids[i];\n        }\n        return this.parseTrades (trades, undefined, since, limit);\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = undefined;\n        try {\n            response = await this.privatePostCancelOrder (this.extend ({\n                'txid': id,\n            }, params));\n        } catch (e) {\n            if (this.last_http_response)\n                if (this.last_http_response.indexOf ('EOrder:Unknown order') >= 0)\n                    throw new OrderNotFound (this.id + ' cancelOrder() error ' + this.last_http_response);\n            throw e;\n        }\n        return response;\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        if (since)\n            request['start'] = parseInt (since / 1000);\n        let response = await this.privatePostOpenOrders (this.extend (request, params));\n        let orders = this.parseOrders (response['result']['open'], undefined, since, limit);\n        return this.filterOrdersBySymbol (orders, symbol);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        if (since)\n            request['start'] = parseInt (since / 1000);\n        let response = await this.privatePostClosedOrders (this.extend (request, params));\n        let orders = this.parseOrders (response['result']['closed'], undefined, since, limit);\n        return this.filterOrdersBySymbol (orders, symbol);\n    }\n\n    async fetchDepositMethods (code = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {};\n        if (code) {\n            let currency = this.currency (code);\n            request['asset'] = currency['id'];\n        }\n        let response = await this.privatePostDepositMethods (this.extend (request, params));\n        return response['result'];\n    }\n\n    async createDepositAddress (currency, params = {}) {\n        let request = {\n            'new': 'true',\n        };\n        let response = await this.fetchDepositAddress (currency, this.extend (request, params));\n        return {\n            'currency': currency,\n            'address': response['address'],\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    async fetchDepositAddress (code, params = {}) {\n        let method = this.safeValue (params, 'method');\n        if (!method)\n            throw new ExchangeError (this.id + ' fetchDepositAddress() requires an extra `method` parameter');\n        await this.loadMarkets ();\n        let currency = this.currency (code);\n        let request = {\n            'asset': currency['id'],\n            'method': method,\n            'new': 'false',\n        };\n        let response = await this.privatePostDepositAddresses (this.extend (request, params));\n        let result = response['result'];\n        let numResults = result.length;\n        if (numResults < 1)\n            throw new ExchangeError (this.id + ' privatePostDepositAddresses() returned no addresses');\n        let address = this.safeString (result[0], 'address');\n        return {\n            'currency': code,\n            'address': address,\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        if ('key' in params) {\n            await this.loadMarkets ();\n            let response = await this.privatePostWithdraw (this.extend ({\n                'asset': currency,\n                'amount': amount,\n                // 'address': address, // they don't allow withdrawals to direct addresses\n            }, params));\n            return {\n                'info': response,\n                'id': response['result'],\n            };\n        }\n        throw new ExchangeError (this.id + \" withdraw requires a 'key' parameter (withdrawal key name, as set up on your account)\");\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = '/' + this.version + '/' + api + '/' + path;\n        if (api === 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            body = this.urlencode (this.extend ({ 'nonce': nonce }, params));\n            let auth = this.encode (nonce + body);\n            let hash = this.hash (auth, 'sha256', 'binary');\n            let binary = this.stringToBinary (this.encode (url));\n            let binhash = this.binaryConcat (binary, hash);\n            let secret = this.base64ToBinary (this.secret);\n            let signature = this.hmac (binhash, secret, 'sha512', 'base64');\n            headers = {\n                'API-Key': this.apiKey,\n                'API-Sign': this.decode (signature),\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        url = this.urls['api'] + url;\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response) {\n            let numErrors = response['error'].length;\n            if (numErrors) {\n                for (let i = 0; i < response['error'].length; i++) {\n                    if (response['error'][i] === 'EService:Unavailable')\n                        throw new ExchangeNotAvailable (this.id + ' ' + this.json (response));\n                    if (response['error'][i] === 'EService:Busy')\n                        throw new DDoSProtection (this.id + ' ' + this.json (response));\n                }\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n            }\n        }\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InvalidNonce, InvalidOrder, AuthenticationError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class kucoin extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'kucoin',\n            'name': 'Kucoin',\n            'countries': 'HK', // Hong Kong\n            'version': 'v1',\n            'rateLimit': 2000,\n            'userAgent': this.userAgents['chrome'],\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n                'fetchOHLCV': true, // see the method implementation below\n                'fetchOrder': false,\n                'fetchOrders': true,\n                'fetchClosedOrders': true,\n                'fetchOpenOrders': true,\n                'fetchMyTrades': false,\n                'fetchCurrencies': true,\n                'withdraw': true,\n            },\n            'timeframes': {\n                '1m': '1',\n                '5m': '5',\n                '15m': '15',\n                '30m': '30',\n                '1h': '60',\n                '8h': '480',\n                '1d': 'D',\n                '1w': 'W',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/33795655-b3c46e48-dcf6-11e7-8abe-dc4588ba7901.jpg',\n                'api': 'https://api.kucoin.com',\n                'www': 'https://kucoin.com',\n                'doc': 'https://kucoinapidocs.docs.apiary.io',\n                'fees': 'https://news.kucoin.com/en/fee',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'open/chart/config',\n                        'open/chart/history',\n                        'open/chart/symbol',\n                        'open/currencies',\n                        'open/deal-orders',\n                        'open/kline',\n                        'open/lang-list',\n                        'open/orders',\n                        'open/orders-buy',\n                        'open/orders-sell',\n                        'open/tick',\n                        'market/open/coin-info',\n                        'market/open/coins',\n                        'market/open/coins-trending',\n                        'market/open/symbols',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'account/balance',\n                        'account/{coin}/wallet/address',\n                        'account/{coin}/wallet/records',\n                        'account/{coin}/balance',\n                        'account/promotion/info',\n                        'account/promotion/sum',\n                        'deal-orders',\n                        'order/active',\n                        'order/active-map',\n                        'order/dealt',\n                        'referrer/descendant/count',\n                        'user/info',\n                    ],\n                    'post': [\n                        'account/{coin}/withdraw/apply',\n                        'account/{coin}/withdraw/cancel',\n                        'cancel-order',\n                        'order',\n                        'user/change-lang',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.0010,\n                    'taker': 0.0010,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'KCS': 2.0,\n                        'BTC': 0.0005,\n                        'USDT': 10.0,\n                        'ETH': 0.01,\n                        'LTC': 0.001,\n                        'NEO': 0.0,\n                        'GAS': 0.0,\n                        'KNC': 0.5,\n                        'BTM': 5.0,\n                        'QTUM': 0.1,\n                        'EOS': 0.5,\n                        'CVC': 3.0,\n                        'OMG': 0.1,\n                        'PAY': 0.5,\n                        'SNT': 20.0,\n                        'BHC': 1.0,\n                        'HSR': 0.01,\n                        'WTC': 0.1,\n                        'VEN': 2.0,\n                        'MTH': 10.0,\n                        'RPX': 1.0,\n                        'REQ': 20.0,\n                        'EVX': 0.5,\n                        'MOD': 0.5,\n                        'NEBL': 0.1,\n                        'DGB': 0.5,\n                        'CAG': 2.0,\n                        'CFD': 0.5,\n                        'RDN': 0.5,\n                        'UKG': 5.0,\n                        'BCPT': 5.0,\n                        'PPT': 0.1,\n                        'BCH': 0.0005,\n                        'STX': 2.0,\n                        'NULS': 1.0,\n                        'GVT': 0.1,\n                        'HST': 2.0,\n                        'PURA': 0.5,\n                        'SUB': 2.0,\n                        'QSP': 5.0,\n                        'POWR': 1.0,\n                        'FLIXX': 10.0,\n                        'LEND': 20.0,\n                        'AMB': 3.0,\n                        'RHOC': 2.0,\n                        'R': 2.0,\n                        'DENT': 50.0,\n                        'DRGN': 1.0,\n                        'ACT': 0.1,\n                    },\n                    'deposit': 0.00,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicGetMarketOpenSymbols ();\n        let markets = response['data'];\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let id = market['symbol'];\n            let base = market['coinType'];\n            let quote = market['coinTypePair'];\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            let precision = {\n                'amount': 8,\n                'price': 8,\n            };\n            let active = market['trading'];\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': active,\n                'info': market,\n                'lot': Math.pow (10, -precision['amount']),\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision['amount']),\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                },\n            }));\n        }\n        return result;\n    }\n\n    async fetchCurrencies (params = {}) {\n        let response = await this.publicGetMarketOpenCoins (params);\n        let currencies = response['data'];\n        let result = {};\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let id = currency['coin'];\n            // todo: will need to rethink the fees\n            // to add support for multiple withdrawal/deposit methods and\n            // differentiated fees for each particular method\n            let code = this.commonCurrencyCode (id);\n            let precision = currency['tradePrecision'];\n            let deposit = currency['enableDeposit'];\n            let withdraw = currency['enableWithdraw'];\n            let active = (deposit && withdraw);\n            result[code] = {\n                'id': id,\n                'code': code,\n                'info': currency,\n                'name': currency['name'],\n                'active': active,\n                'status': 'ok',\n                'fee': currency['withdrawFeeRate'], // todo: redesign\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': currency['withdrawMinAmount'],\n                        'max': Math.pow (10, precision),\n                    },\n                },\n            };\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetAccountBalance (this.extend ({\n            'limit': 20, // default 12, max 20\n            'page': 1,\n        }, params));\n        let balances = response['data'];\n        let result = { 'info': balances };\n        let indexed = this.indexBy (balances, 'coinType');\n        let keys = Object.keys (indexed);\n        for (let i = 0; i < keys.length; i++) {\n            let id = keys[i];\n            let currency = this.commonCurrencyCode (id);\n            let account = this.account ();\n            let balance = indexed[id];\n            let used = parseFloat (balance['freezeBalance']);\n            let free = parseFloat (balance['balance']);\n            let total = this.sum (free, used);\n            account['free'] = free;\n            account['used'] = used;\n            account['total'] = total;\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetOpenOrders (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        let orderbook = response['data'];\n        return this.parseOrderBook (orderbook, undefined, 'BUY', 'SELL');\n    }\n\n    parseOrder (order, market = undefined) {\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else {\n            symbol = order['coinType'] + '/' + order['coinTypePair'];\n        }\n        let timestamp = order['createdAt'];\n        let price = this.safeValue (order, 'price');\n        if (typeof price === 'undefined')\n            price = this.safeValue (order, 'dealPrice');\n        let amount = this.safeValue (order, 'amount');\n        let filled = this.safeValue (order, 'dealAmount', 0);\n        let remaining = this.safeValue (order, 'pendingAmount');\n        if (typeof amount === 'undefined')\n            if (typeof filled !== 'undefined')\n                if (typeof remaining !== 'undefined')\n                    amount = this.sum (filled, remaining);\n        let side = order['direction'].toLowerCase ();\n        let fee = undefined;\n        if ('fee' in order) {\n            fee = {\n                'cost': this.safeFloat (order, 'fee'),\n                'rate': this.safeFloat (order, 'feeRate'),\n            };\n            if (market)\n                fee['currency'] = market['base'];\n        }\n        let status = this.safeValue (order, 'status');\n        let result = {\n            'info': order,\n            'id': this.safeString (order, 'oid'),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': 'limit',\n            'side': side,\n            'price': price,\n            'amount': amount,\n            'cost': price * filled,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': fee,\n        };\n        return result;\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOpenOrders requires a symbol param');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n        };\n        let response = await this.privateGetOrderActiveMap (this.extend (request, params));\n        let orders = this.arrayConcat (response['data']['SELL'], response['data']['BUY']);\n        let result = [];\n        for (let i = 0; i < orders.length; i++) {\n            result.push (this.extend (orders[i], { 'status': 'open' }));\n        }\n        return this.parseOrders (result, market, since, limit);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let request = {};\n        await this.loadMarkets ();\n        let market = undefined;\n        if (symbol) {\n            market = this.market (symbol);\n            request['symbol'] = market['id'];\n        }\n        if (since) {\n            request['since'] = since;\n        }\n        if (limit) {\n            request['limit'] = limit;\n        }\n        let response = await this.privateGetOrderDealt (this.extend (request, params));\n        let orders = response['data']['datas'];\n        let result = [];\n        for (let i = 0; i < orders.length; i++) {\n            result.push (this.extend (orders[i], { 'status': 'closed' }));\n        }\n        return this.parseOrders (result, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type !== 'limit')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let base = market['base'];\n        let order = {\n            'symbol': market['id'],\n            'type': side.toUpperCase (),\n            'price': this.priceToPrecision (symbol, price),\n            'amount': this.truncate (amount, this.currencies[base]['precision']),\n        };\n        let response = await this.privatePostOrder (this.extend (order, params));\n        return {\n            'info': response,\n            'id': this.safeString (response['data'], 'orderOid'),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' cancelOrder requires symbol argument');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n            'orderOid': id,\n        };\n        if ('type' in params) {\n            request['type'] = params['type'].toUpperCase ();\n        } else {\n            throw new ExchangeError (this.id + ' cancelOrder requires type (BUY or SELL) param');\n        }\n        let response = await this.privatePostCancelOrder (this.extend (request, params));\n        return response;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['datetime'];\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        } else {\n            symbol = ticker['coinType'] + '/' + ticker['coinTypePair'];\n        }\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'buy'),\n            'ask': this.safeFloat (ticker, 'sell'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'lastDealPrice'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'vol'),\n            'quoteVolume': this.safeFloat (ticker, 'volValue'),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        let response = await this.publicGetMarketOpenSymbols (params);\n        let tickers = response['data'];\n        let result = {};\n        for (let t = 0; t < tickers.length; t++) {\n            let ticker = this.parseTicker (tickers[t]);\n            let symbol = ticker['symbol'];\n            result[symbol] = ticker;\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetOpenTick (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        let ticker = response['data'];\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = trade[0];\n        let side = undefined;\n        if (trade[1] === 'BUY') {\n            side = 'buy';\n        } else if (trade[1] === 'SELL') {\n            side = 'sell';\n        }\n        return {\n            'id': undefined,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': 'limit',\n            'side': side,\n            'price': trade[2],\n            'amount': trade[3],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetOpenDealOrders (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTrades (response['data'], market, since, limit);\n    }\n\n    parseTradingViewOHLCVs (ohlcvs, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        let result = [];\n        for (let i = 0; i < ohlcvs['t'].length; i++) {\n            result.push ([\n                ohlcvs['t'][i],\n                ohlcvs['o'][i],\n                ohlcvs['h'][i],\n                ohlcvs['l'][i],\n                ohlcvs['c'][i],\n                ohlcvs['v'][i],\n            ]);\n        }\n        return this.parseOHLCVs (result, market, timeframe, since, limit);\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let end = this.seconds ();\n        let resolution = this.timeframes[timeframe];\n        // convert 'resolution' to minutes in order to calculate 'from' later\n        let minutes = resolution;\n        if (minutes === 'D') {\n            if (!limit)\n                limit = 30; // 30 days, 1 month\n            minutes = 1440;\n        } else if (minutes === 'W') {\n            if (!limit)\n                limit = 52; // 52 weeks, 1 year\n            minutes = 10080;\n        } else if (!limit) {\n            limit = 1440;\n            minutes = 1440;\n        }\n        let start = end - minutes * 60 * limit;\n        if (since) {\n            start = parseInt (since / 1000);\n            end = this.sum (start, minutes * 60 * limit);\n        }\n        let request = {\n            'symbol': market['id'],\n            'type': this.timeframes[timeframe],\n            'resolution': resolution,\n            'from': start,\n            'to': end,\n        };\n        let response = await this.publicGetOpenChartHistory (this.extend (request, params));\n        return this.parseTradingViewOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    async withdraw (code, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let currency = this.currency (code);\n        let response = await this.privatePostAccountCoinWithdrawApply (this.extend ({\n            'coin': currency['id'],\n            'amount': amount,\n            'address': address,\n        }, params));\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let endpoint = '/' + this.version + '/' + this.implodeParams (path, params);\n        let url = this.urls['api'] + endpoint;\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            // their nonce is always a calibrated synched milliseconds-timestamp\n            let nonce = this.milliseconds ();\n            let queryString = '';\n            nonce = nonce.toString ();\n            if (Object.keys (query).length) {\n                queryString = this.rawencode (this.keysort (query));\n                url += '?' + queryString;\n                if (method !== 'GET') {\n                    body = queryString;\n                }\n            }\n            let auth = endpoint + '/' + nonce + '/' + queryString;\n            let payload = this.stringToBase64 (this.encode (auth));\n            // payload should be \"encoded\" as returned from stringToBase64\n            let signature = this.hmac (payload, this.encode (this.secret), 'sha256');\n            headers = {\n                'KC-API-KEY': this.apiKey,\n                'KC-API-NONCE': nonce,\n                'KC-API-SIGNATURE': signature,\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    throwExceptionOnError (response) {\n        if ('success' in response) {\n            if (!response['success']) {\n                if ('code' in response) {\n                    let message = this.safeString (response, 'msg');\n                    if (response['code'] === 'UNAUTH') {\n                        if (message === 'Invalid nonce')\n                            throw new InvalidNonce (this.id + ' ' + message);\n                        throw new AuthenticationError (this.id + ' ' + this.json (response));\n                    } else if (response['code'] === 'ERROR') {\n                        if (message.indexOf ('precision of amount') >= 0)\n                            throw new InvalidOrder (this.id + ' ' + message);\n                        if (message.indexOf ('Min amount each order') >= 0)\n                            throw new InvalidOrder (this.id + ' ' + message);\n                    }\n                }\n            }\n        }\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (body && (body[0] === '{')) {\n            let response = JSON.parse (body);\n            this.throwExceptionOnError (response);\n        }\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        this.throwExceptionOnError (response);\n        return response;\n    }\n}\n","'use strict';\n\n// ---------------------------------------------------------------------------\n\nconst acx = require ('./acx.js');\nconst { ExchangeError, InsufficientFunds, OrderNotFound } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class kuna extends acx {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'kuna',\n            'name': 'Kuna',\n            'countries': 'UA',\n            'rateLimit': 1000,\n            'version': 'v2',\n            'has': {\n                'CORS': false,\n                'fetchTickers': false,\n                'fetchOHLCV': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/31697638-912824fa-b3c1-11e7-8c36-cf9606eb94ac.jpg',\n                'api': 'https://kuna.io',\n                'www': 'https://kuna.io',\n                'doc': 'https://kuna.io/documents/api',\n                'fees': 'https://kuna.io/documents/api',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'tickers/{market}',\n                        'order_book',\n                        'order_book/{market}',\n                        'trades',\n                        'trades/{market}',\n                        'timestamp',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'members/me',\n                        'orders',\n                        'trades/my',\n                    ],\n                    'post': [\n                        'orders',\n                        'order/delete',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/UAH': { 'id': 'btcuah', 'symbol': 'BTC/UAH', 'base': 'BTC', 'quote': 'UAH', 'precision': { 'amount': 6, 'price': 0 }, 'lot': 0.000001, 'limits': { 'amount': { 'min': 0.000001, 'max': undefined }, 'price': { 'min': 1, 'max': undefined }, 'cost': { 'min': 0.000001, 'max': undefined }}},\n                'ETH/UAH': { 'id': 'ethuah', 'symbol': 'ETH/UAH', 'base': 'ETH', 'quote': 'UAH', 'precision': { 'amount': 6, 'price': 0 }, 'lot': 0.000001, 'limits': { 'amount': { 'min': 0.000001, 'max': undefined }, 'price': { 'min': 1, 'max': undefined }, 'cost': { 'min': 0.000001, 'max': undefined }}},\n                'GBG/UAH': { 'id': 'gbguah', 'symbol': 'GBG/UAH', 'base': 'GBG', 'quote': 'UAH', 'precision': { 'amount': 3, 'price': 2 }, 'lot': 0.001, 'limits': { 'amount': { 'min': 0.000001, 'max': undefined }, 'price': { 'min': 0.01, 'max': undefined }, 'cost': { 'min': 0.000001, 'max': undefined }}}, // Golos Gold (GBG != GOLOS)\n                'KUN/BTC': { 'id': 'kunbtc', 'symbol': 'KUN/BTC', 'base': 'KUN', 'quote': 'BTC', 'precision': { 'amount': 6, 'price': 6 }, 'lot': 0.000001, 'limits': { 'amount': { 'min': 0.000001, 'max': undefined }, 'price': { 'min': 0.000001, 'max': undefined }, 'cost': { 'min': 0.000001, 'max': undefined }}},\n                'BCH/BTC': { 'id': 'bchbtc', 'symbol': 'BCH/BTC', 'base': 'BCH', 'quote': 'BTC', 'precision': { 'amount': 6, 'price': 6 }, 'lot': 0.000001, 'limits': { 'amount': { 'min': 0.000001, 'max': undefined }, 'price': { 'min': 0.000001, 'max': undefined }, 'cost': { 'min': 0.000001, 'max': undefined }}},\n                'BCH/UAH': { 'id': 'bchuah', 'symbol': 'BCH/UAH', 'base': 'BCH', 'quote': 'UAH', 'precision': { 'amount': 6, 'price': 0 }, 'lot': 0.000001, 'limits': { 'amount': { 'min': 0.000001, 'max': undefined }, 'price': { 'min': 1, 'max': undefined }, 'cost': { 'min': 0.000001, 'max': undefined }}},\n                'WAVES/UAH': { 'id': 'wavesuah', 'symbol': 'WAVES/UAH', 'base': 'WAVES', 'quote': 'UAH', 'precision': { 'amount': 6, 'price': 0 }, 'lot': 0.000001, 'limits': { 'amount': { 'min': 0.000001, 'max': undefined }, 'price': { 'min': 1, 'max': undefined }, 'cost': { 'min': 0.000001, 'max': undefined }}},\n                'ARN/BTC': { 'id': 'arnbtc', 'symbol': 'ARN/BTC', 'base': 'ARN', 'quote': 'BTC' },\n                'B2B/BTC': { 'id': 'b2bbtc', 'symbol': 'B2B/BTC', 'base': 'B2B', 'quote': 'BTC' },\n                'EVR/BTC': { 'id': 'evrbtc', 'symbol': 'EVR/BTC', 'base': 'EVR', 'quote': 'BTC' },\n                'GOL/GBG': { 'id': 'golgbg', 'symbol': 'GOL/GBG', 'base': 'GOL', 'quote': 'GBG' },\n                'R/BTC': { 'id': 'rbtc', 'symbol': 'R/BTC', 'base': 'R', 'quote': 'BTC' },\n                'RMC/BTC': { 'id': 'rmcbtc', 'symbol': 'RMC/BTC', 'base': 'RMC', 'quote': 'BTC' },\n            },\n            'fees': {\n                'trading': {\n                    'taker': 0.25 / 100,\n                    'maker': 0.25 / 100,\n                },\n                'funding': {\n                    'withdraw': {\n                        'UAH': '1%',\n                        'BTC': 0.001,\n                        'BCH': 0.001,\n                        'ETH': 0.01,\n                        'WAVES': 0.01,\n                        'GOL': 0.0,\n                        'GBG': 0.0,\n                        // 'RMC': 0.001 BTC\n                        // 'ARN': 0.01 ETH\n                        // 'R': 0.01 ETH\n                        // 'EVR': 0.01 ETH\n                    },\n                    'deposit': {\n                        // 'UAH': (amount) => amount * 0.001 + 5\n                    },\n                },\n            },\n        });\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code === 400) {\n            let response = JSON.parse (body);\n            let error = this.safeValue (response, 'error');\n            let errorCode = this.safeInteger (error, 'code');\n            if (errorCode === 2002) {\n                throw new InsufficientFunds ([ this.id, method, url, code, reason, body ].join (' '));\n            } else if (errorCode === 2003) {\n                throw new OrderNotFound ([ this.id, method, url, code, reason, body ].join (' '));\n            }\n        }\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let market = this.market (symbol);\n        let orderBook = await this.publicGetOrderBook (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseOrderBook (orderBook, undefined, 'bids', 'asks', 'price', 'remaining_volume');\n    }\n\n    async fetchL3OrderBook (symbol, params) {\n        return this.fetchOrderBook (symbol, params);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOpenOrders requires a symbol argument');\n        let market = this.market (symbol);\n        let orders = await this.privateGetOrders (this.extend ({\n            'market': market['id'],\n        }, params));\n        // todo emulation of fetchClosedOrders, fetchOrders, fetchOrder\n        // with order cache + fetchOpenOrders\n        // as in BTC-e, Liqui, Yobit, DSX, Tidex, WEX\n        return this.parseOrders (orders, market, since, limit);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = this.parse8601 (trade['created_at']);\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'id': trade['id'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': undefined,\n            'side': undefined,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['volume']),\n            'info': trade,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetTrades (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    parseMyTrade (trade, market) {\n        let timestamp = this.parse8601 (trade['created_at']);\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'id': trade['id'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'price': trade['price'],\n            'amount': trade['volume'],\n            'cost': trade['funds'],\n            'symbol': symbol,\n            'side': trade['side'],\n            'order': trade['order_id'],\n        };\n    }\n\n    parseMyTrades (trades, market = undefined) {\n        let parsedTrades = [];\n        for (let i = 0; i < trades.length; i++) {\n            let trade = trades[i];\n            let parsedTrade = this.parseMyTrade (trade, market);\n            parsedTrades.push (parsedTrade);\n        }\n        return parsedTrades;\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOpenOrders requires a symbol argument');\n        let market = this.market (symbol);\n        let response = await this.privateGetTradesMy ({ 'market': market['id'] });\n        return this.parseMyTrades (response, market);\n    }\n};\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class lakebtc extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'lakebtc',\n            'name': 'LakeBTC',\n            'countries': 'US',\n            'version': 'api_v2',\n            'has': {\n                'CORS': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28074120-72b7c38a-6660-11e7-92d9-d9027502281d.jpg',\n                'api': 'https://api.lakebtc.com',\n                'www': 'https://www.lakebtc.com',\n                'doc': [\n                    'https://www.lakebtc.com/s/api_v2',\n                    'https://www.lakebtc.com/s/api',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'bcorderbook',\n                        'bctrades',\n                        'ticker',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'buyOrder',\n                        'cancelOrders',\n                        'getAccountInfo',\n                        'getExternalAccounts',\n                        'getOrders',\n                        'getTrades',\n                        'openOrders',\n                        'sellOrder',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.15 / 100,\n                    'taker': 0.2 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetTicker ();\n        let result = [];\n        let keys = Object.keys (markets);\n        for (let k = 0; k < keys.length; k++) {\n            let id = keys[k];\n            let market = markets[id];\n            let base = id.slice (0, 3);\n            let quote = id.slice (3, 6);\n            base = base.toUpperCase ();\n            quote = quote.toUpperCase ();\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetAccountInfo ();\n        let balances = response['balance'];\n        let result = { 'info': response };\n        let currencies = Object.keys (balances);\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let balance = parseFloat (balances[currency]);\n            let account = {\n                'free': balance,\n                'used': 0.0,\n                'total': balance,\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetBcorderbook (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let tickers = await this.publicGetTicker (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        let ticker = tickers[market['id']];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'bid'),\n            'ask': this.safeFloat (ticker, 'ask'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'volume'),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['date'] * 1000;\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': trade['tid'].toString (),\n            'order': undefined,\n            'type': undefined,\n            'side': undefined,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetBctrades (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (market, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        if (type == 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let method = 'privatePost' + this.capitalize (side) + 'Order';\n        let marketId = this.marketId (market);\n        let order = {\n            'params': [ price, amount, marketId ],\n        };\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelOrder ({ 'params': id });\n    }\n\n    nonce () {\n        return this.microseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version;\n        if (api == 'public') {\n            url += '/' + path;\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            if (Object.keys (params).length)\n                params = params.join (',');\n            else\n                params = '';\n            let query = this.urlencode ({\n                'tonce': nonce,\n                'accesskey': this.apiKey,\n                'requestmethod': method.toLowerCase (),\n                'id': nonce,\n                'method': path,\n                'params': params,\n            });\n            body = this.json ({\n                'method': path,\n                'params': params,\n                'id': nonce,\n            });\n            let signature = this.hmac (this.encode (query), this.encode (this.secret), 'sha1');\n            let auth = this.encode (this.apiKey + ':' + signature);\n            headers = {\n                'Json-Rpc-Tonce': nonce,\n                'Authorization': \"Basic \" + this.decode (this.stringToBase64 (auth)),\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","'use strict';\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InsufficientFunds, OrderNotFound, DDoSProtection, InvalidOrder, AuthenticationError } = require ('./base/errors');\n\nmodule.exports = class liqui extends Exchange {\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'liqui',\n            'name': 'Liqui',\n            'countries': 'UA',\n            'rateLimit': 3000,\n            'version': '3',\n            'userAgent': this.userAgents['chrome'],\n            'has': {\n                'CORS': false,\n                'fetchOrder': true,\n                'fetchOrders': 'emulated',\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': 'emulated',\n                'fetchTickers': true,\n                'fetchMyTrades': true,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27982022-75aea828-63a0-11e7-9511-ca584a8edd74.jpg',\n                'api': {\n                    'public': 'https://api.liqui.io/api',\n                    'private': 'https://api.liqui.io/tapi',\n                },\n                'www': 'https://liqui.io',\n                'doc': 'https://liqui.io/api',\n                'fees': 'https://liqui.io/fee',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'info',\n                        'ticker/{pair}',\n                        'depth/{pair}',\n                        'trades/{pair}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'getInfo',\n                        'Trade',\n                        'ActiveOrders',\n                        'OrderInfo',\n                        'CancelOrder',\n                        'TradeHistory',\n                        'CoinDepositAddress',\n                        'WithdrawCoin',\n                        'CreateCoupon',\n                        'RedeemCoupon',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.001,\n                    'taker': 0.0025,\n                },\n                'funding': 0.0,\n            },\n            'exceptions': {\n                '803': InvalidOrder, // \"Count could not be less than 0.001.\" (selling below minAmount)\n                '804': InvalidOrder, // \"Count could not be more than 10000.\" (buying above maxAmount)\n                '805': InvalidOrder, // \"price could not be less than X.\" (minPrice violation on buy & sell)\n                '806': InvalidOrder, // \"price could not be more than X.\" (maxPrice violation on buy & sell)\n                '807': InvalidOrder, // \"cost could not be less than X.\" (minCost violation on buy & sell)\n                '831': InsufficientFunds, // \"Not enougth X to create buy order.\" (buying with balance.quote < order.cost)\n                '832': InsufficientFunds, // \"Not enougth X to create sell order.\" (selling with balance.base < order.amount)\n                '833': OrderNotFound, // \"Order with id X was not found.\" (cancelling non-existent, closed and cancelled order)\n            },\n        });\n    }\n\n    calculateFee (symbol, type, side, amount, price, takerOrMaker = 'taker', params = {}) {\n        let market = this.markets[symbol];\n        let key = 'quote';\n        let rate = market[takerOrMaker];\n        let cost = parseFloat (this.costToPrecision (symbol, amount * rate));\n        if (side === 'sell') {\n            cost *= price;\n        } else {\n            key = 'base';\n        }\n        return {\n            'type': takerOrMaker,\n            'currency': market[key],\n            'rate': rate,\n            'cost': cost,\n        };\n    }\n\n    commonCurrencyCode (currency) {\n        if (!this.substituteCommonCurrencyCodes)\n            return currency;\n        if (currency === 'XBT')\n            return 'BTC';\n        if (currency === 'BCC')\n            return 'BCH';\n        if (currency === 'DRK')\n            return 'DASH';\n        // they misspell DASH as dsh :/\n        if (currency === 'DSH')\n            return 'DASH';\n        return currency;\n    }\n\n    getBaseQuoteFromMarketId (id) {\n        let uppercase = id.toUpperCase ();\n        let [ base, quote ] = uppercase.split ('_');\n        base = this.commonCurrencyCode (base);\n        quote = this.commonCurrencyCode (quote);\n        return [ base, quote ];\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicGetInfo ();\n        let markets = response['pairs'];\n        let keys = Object.keys (markets);\n        let result = [];\n        for (let p = 0; p < keys.length; p++) {\n            let id = keys[p];\n            let market = markets[id];\n            let [ base, quote ] = this.getBaseQuoteFromMarketId (id);\n            let symbol = base + '/' + quote;\n            let precision = {\n                'amount': this.safeInteger (market, 'decimal_places'),\n                'price': this.safeInteger (market, 'decimal_places'),\n            };\n            let amountLimits = {\n                'min': this.safeFloat (market, 'min_amount'),\n                'max': this.safeFloat (market, 'max_amount'),\n            };\n            let priceLimits = {\n                'min': this.safeFloat (market, 'min_price'),\n                'max': this.safeFloat (market, 'max_price'),\n            };\n            let costLimits = {\n                'min': this.safeFloat (market, 'min_total'),\n            };\n            let limits = {\n                'amount': amountLimits,\n                'price': priceLimits,\n                'cost': costLimits,\n            };\n            let hidden = this.safeInteger (market, 'hidden');\n            let active = (hidden === 0);\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': active,\n                'taker': market['fee'] / 100,\n                'lot': amountLimits['min'],\n                'precision': precision,\n                'limits': limits,\n                'info': market,\n            }));\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetInfo ();\n        let balances = response['return'];\n        let result = { 'info': balances };\n        let funds = balances['funds'];\n        let currencies = Object.keys (funds);\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let uppercase = currency.toUpperCase ();\n            uppercase = this.commonCurrencyCode (uppercase);\n            let total = undefined;\n            let used = undefined;\n            if (balances['open_orders'] === 0) {\n                total = funds[currency];\n                used = 0.0;\n            }\n            let account = {\n                'free': funds[currency],\n                'used': used,\n                'total': total,\n            };\n            result[uppercase] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetDepthPair (this.extend ({\n            'pair': market['id'],\n            // 'limit': 150, // default = 150, max = 2000\n        }, params));\n        let market_id_in_reponse = (market['id'] in response);\n        if (!market_id_in_reponse)\n            throw new ExchangeError (this.id + ' ' + market['symbol'] + ' order book is empty or not available');\n        let orderbook = response[market['id']];\n        let result = this.parseOrderBook (orderbook);\n        result['bids'] = this.sortBy (result['bids'], 0, true);\n        result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['updated'] * 1000;\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'buy'),\n            'ask': this.safeFloat (ticker, 'sell'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': this.safeFloat (ticker, 'avg'),\n            'baseVolume': this.safeFloat (ticker, 'vol_cur'),\n            'quoteVolume': this.safeFloat (ticker, 'vol'),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let ids = undefined;\n        if (!symbols) {\n            // let numIds = this.ids.length;\n            // if (numIds > 256)\n            //     throw new ExchangeError (this.id + ' fetchTickers() requires symbols argument');\n            ids = this.ids.join ('-');\n            if (ids.length > 2083) {\n                let numIds = this.ids.length;\n                throw new ExchangeError (this.id + ' has ' + numIds.toString () + ' symbols exceeding max URL length, you are required to specify a list of symbols in the first argument to fetchTickers');\n            }\n        } else {\n            ids = this.marketIds (symbols);\n            ids = ids.join ('-');\n        }\n        let tickers = await this.publicGetTickerPair (this.extend ({\n            'pair': ids,\n        }, params));\n        let result = {};\n        let keys = Object.keys (tickers);\n        for (let k = 0; k < keys.length; k++) {\n            let id = keys[k];\n            let ticker = tickers[id];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let tickers = await this.fetchTickers ([ symbol ], params);\n        return tickers[symbol];\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = parseInt (trade['timestamp']) * 1000;\n        let side = trade['type'];\n        if (side === 'ask')\n            side = 'sell';\n        if (side === 'bid')\n            side = 'buy';\n        let price = this.safeFloat (trade, 'price');\n        if ('rate' in trade)\n            price = this.safeFloat (trade, 'rate');\n        let id = this.safeString (trade, 'tid');\n        if ('trade_id' in trade)\n            id = this.safeString (trade, 'trade_id');\n        let order = this.safeString (trade, this.getOrderIdKey ());\n        if ('pair' in trade) {\n            let marketId = trade['pair'];\n            market = this.markets_by_id[marketId];\n        }\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let amount = trade['amount'];\n        let type = 'limit'; // all trades are still limit trades\n        let fee = undefined;\n        // this is filled by fetchMyTrades() only\n        // is_your_order is always false :\\\n        // let isYourOrder = this.safeValue (trade, 'is_your_order');\n        // let takerOrMaker = 'taker';\n        // if (isYourOrder)\n        //     takerOrMaker = 'maker';\n        // let fee = this.calculateFee (symbol, type, side, amount, price, takerOrMaker);\n        return {\n            'id': id,\n            'order': order,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': price,\n            'amount': amount,\n            'fee': fee,\n            'info': trade,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'pair': market['id'],\n        };\n        if (limit)\n            request['limit'] = limit;\n        let response = await this.publicGetTradesPair (this.extend (request, params));\n        return this.parseTrades (response[market['id']], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type === 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'pair': market['id'],\n            'type': side,\n            'amount': this.amountToPrecision (symbol, amount),\n            'rate': this.priceToPrecision (symbol, price),\n        };\n        let response = await this.privatePostTrade (this.extend (request, params));\n        let id = this.safeString (response['return'], this.getOrderIdKey ());\n        let timestamp = this.milliseconds ();\n        price = parseFloat (price);\n        amount = parseFloat (amount);\n        let status = 'open';\n        if (id === '0') {\n            id = this.safeString (response['return'], 'init_order_id');\n            status = 'closed';\n        }\n        let filled = this.safeFloat (response['return'], 'received', 0.0);\n        let remaining = this.safeFloat (response['return'], 'remains', amount);\n        let order = {\n            'id': id,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': status,\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': price,\n            'cost': price * filled,\n            'amount': amount,\n            'remaining': remaining,\n            'filled': filled,\n            'fee': undefined,\n            // 'trades': this.parseTrades (order['trades'], market),\n        };\n        this.orders[id] = order;\n        return this.extend ({ 'info': response }, order);\n    }\n\n    getOrderIdKey () {\n        return 'order_id';\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = undefined;\n        let request = {};\n        let idKey = this.getOrderIdKey ();\n        request[idKey] = id;\n        response = await this.privatePostCancelOrder (this.extend (request, params));\n        if (id in this.orders)\n            this.orders[id]['status'] = 'canceled';\n        return response;\n    }\n\n    parseOrder (order, market = undefined) {\n        let id = order['id'].toString ();\n        let status = this.safeInteger (order, 'status');\n        if (status === 0) {\n            status = 'open';\n        } else if (status === 1) {\n            status = 'closed';\n        } else if ((status === 2) || (status === 3)) {\n            status = 'canceled';\n        }\n        let timestamp = parseInt (order['timestamp_created']) * 1000;\n        let symbol = undefined;\n        if (!market)\n            market = this.markets_by_id[order['pair']];\n        if (market)\n            symbol = market['symbol'];\n        let remaining = undefined;\n        let amount = undefined;\n        let price = this.safeFloat (order, 'rate');\n        let filled = undefined;\n        let cost = undefined;\n        if ('start_amount' in order) {\n            amount = this.safeFloat (order, 'start_amount');\n            remaining = this.safeFloat (order, 'amount');\n        } else {\n            remaining = this.safeFloat (order, 'amount');\n            if (id in this.orders)\n                amount = this.orders[id]['amount'];\n        }\n        if (typeof amount !== 'undefined') {\n            if (typeof remaining !== 'undefined') {\n                filled = amount - remaining;\n                cost = price * filled;\n            }\n        }\n        let fee = undefined;\n        let result = {\n            'info': order,\n            'id': id,\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'type': 'limit',\n            'side': order['type'],\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'remaining': remaining,\n            'filled': filled,\n            'status': status,\n            'fee': fee,\n        };\n        return result;\n    }\n\n    parseOrders (orders, market = undefined, since = undefined, limit = undefined) {\n        let ids = Object.keys (orders);\n        let result = [];\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let order = orders[id];\n            let extended = this.extend (order, { 'id': id });\n            result.push (this.parseOrder (extended, market));\n        }\n        return this.filterBySinceLimit (result, since, limit);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostOrderInfo (this.extend ({\n            'order_id': parseInt (id),\n        }, params));\n        id = id.toString ();\n        let newOrder = this.parseOrder (this.extend ({ 'id': id }, response['return'][id]));\n        let oldOrder = (id in this.orders) ? this.orders[id] : {};\n        this.orders[id] = this.extend (oldOrder, newOrder);\n        return this.orders[id];\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        // if (!symbol)\n        //     throw new ExchangeError (this.id + ' fetchOrders requires a symbol');\n        await this.loadMarkets ();\n        let request = {};\n        let market = undefined;\n        if (symbol) {\n            let market = this.market (symbol);\n            request['pair'] = market['id'];\n        }\n        let response = await this.privatePostActiveOrders (this.extend (request, params));\n        let openOrders = [];\n        if ('return' in response)\n            openOrders = this.parseOrders (response['return'], market);\n        for (let j = 0; j < openOrders.length; j++) {\n            this.orders[openOrders[j]['id']] = openOrders[j];\n        }\n        let openOrdersIndexedById = this.indexBy (openOrders, 'id');\n        let cachedOrderIds = Object.keys (this.orders);\n        let result = [];\n        for (let k = 0; k < cachedOrderIds.length; k++) {\n            let id = cachedOrderIds[k];\n            if (id in openOrdersIndexedById) {\n                this.orders[id] = this.extend (this.orders[id], openOrdersIndexedById[id]);\n            } else {\n                let order = this.orders[id];\n                if (order['status'] === 'open') {\n                    this.orders[id] = this.extend (order, {\n                        'status': 'closed',\n                        'cost': order['amount'] * order['price'],\n                        'filled': order['amount'],\n                        'remaining': 0.0,\n                    });\n                }\n            }\n            let order = this.orders[id];\n            if (symbol) {\n                if (order['symbol'] === symbol)\n                    result.push (order);\n            } else {\n                result.push (order);\n            }\n        }\n        return this.filterBySinceLimit (result, since, limit);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, since, limit, params);\n        let result = [];\n        for (let i = 0; i < orders.length; i++) {\n            if (orders[i]['status'] === 'open')\n                result.push (orders[i]);\n        }\n        return result;\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, since, limit, params);\n        let result = [];\n        for (let i = 0; i < orders.length; i++) {\n            if (orders[i]['status'] === 'closed')\n                result.push (orders[i]);\n        }\n        return result;\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        let request = {\n            // 'from': 123456789, // trade ID, from which the display starts numerical 0\n            // 'count': 1000, // the number of trades for display numerical, default = 1000\n            // 'from_id': trade ID, from which the display starts numerical 0\n            // 'end_id': trade ID on which the display ends numerical ∞\n            // 'order': 'ASC', // sorting, default = DESC\n            // 'since': 1234567890, // UTC start time, default = 0\n            // 'end': 1234567890, // UTC end time, default = ∞\n            // 'pair': 'eth_btc', // default = all markets\n        };\n        if (symbol) {\n            market = this.market (symbol);\n            request['pair'] = market['id'];\n        }\n        if (limit)\n            request['count'] = parseInt (limit);\n        if (since)\n            request['since'] = parseInt (since / 1000);\n        let response = await this.privatePostTradeHistory (this.extend (request, params));\n        let trades = [];\n        if ('return' in response)\n            trades = response['return'];\n        return this.parseTrades (trades, market, since, limit);\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostWithdrawCoin (this.extend ({\n            'coinName': currency,\n            'amount': parseFloat (amount),\n            'address': address,\n        }, params));\n        return {\n            'info': response,\n            'id': response['return']['tId'],\n        };\n    }\n\n    signBodyWithSecret (body) {\n        return this.hmac (this.encode (body), this.encode (this.secret), 'sha512');\n    }\n\n    getVersionString () {\n        return '/' + this.version;\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({\n                'nonce': nonce,\n                'method': path,\n            }, query));\n            let signature = this.signBodyWithSecret (body);\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Key': this.apiKey,\n                'Sign': signature,\n            };\n        } else {\n            url += this.getVersionString () + '/' + this.implodeParams (path, params);\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (httpCode, reason, url, method, headers, body) {\n        if ((typeof body !== 'string') || (body.length < 2))\n            return; // fallback to default error handler\n        if ((body[0] === '{') || (body[0] === '[')) {\n            let response = JSON.parse (body);\n            if ('success' in response) {\n                //\n                // 1 - Liqui only returns the integer 'success' key from their private API\n                //\n                //     { \"success\": 1, ... } httpCode === 200\n                //     { \"success\": 0, ... } httpCode === 200\n                //\n                // 2 - However, exchanges derived from Liqui, can return non-integers\n                //\n                //     It can be a numeric string\n                //     { \"sucesss\": \"1\", ... }\n                //     { \"sucesss\": \"0\", ... }, httpCode >= 200 (can be 403, 502, etc)\n                //\n                //     Or just a string\n                //     { \"success\": \"true\", ... }\n                //     { \"success\": \"false\", ... }, httpCode >= 200\n                //\n                //     Or a boolean\n                //     { \"success\": true, ... }\n                //     { \"success\": false, ... }, httpCode >= 200\n                //\n                // 3 - Oversimplified, Python PEP8 forbids comparison operator (===) of different types\n                //\n                // 4 - We do not want to copy-paste and duplicate the code of this handler to other exchanges derived from Liqui\n                //\n                // To cover points 1, 2, 3 and 4 combined this handler should work like this:\n                //\n                let success = this.safeValue (response, 'success', false);\n                if (typeof success === 'string') {\n                    if ((success === 'true') || (success === '1'))\n                        success = true;\n                    else\n                        success = false;\n                }\n                if (!success) {\n                    const code = response['code'];\n                    const message = response['error'];\n                    const feedback = this.id + ' ' + this.json (response);\n                    const exceptions = this.exceptions;\n                    if (code in exceptions) {\n                        throw new exceptions[code] (feedback);\n                    }\n                    // need a second error map for these messages, apparently...\n                    // in fact, we can use the same .exceptions with string-keys to save some loc here\n                    if (message === 'invalid api key') {\n                        throw new AuthenticationError (feedback);\n                    } else if (message === 'api key dont have trade permission') {\n                        throw new AuthenticationError (feedback);\n                    } else if (message.indexOf ('invalid parameter') >= 0) { // errorCode 0, returned on buy(symbol, 0, 0)\n                        throw new InvalidOrder (feedback);\n                    } else if (message === 'Requests too often') {\n                        throw new DDoSProtection (feedback);\n                    } else if (message === 'not available') {\n                        throw new DDoSProtection (feedback);\n                    } else if (message === 'external service unavailable') {\n                        throw new DDoSProtection (feedback);\n                    } else {\n                        throw new ExchangeError (this.id + ' unknown \"error\" value: ' + this.json (response));\n                    }\n                }\n            }\n        }\n    }\n};\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, AuthenticationError, NotSupported, InvalidOrder, OrderNotFound, ExchangeNotAvailable } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class livecoin extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'livecoin',\n            'name': 'LiveCoin',\n            'countries': [ 'US', 'UK', 'RU' ],\n            'rateLimit': 1000,\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n                'fetchCurrencies': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27980768-f22fc424-638a-11e7-89c9-6010a54ff9be.jpg',\n                'api': 'https://api.livecoin.net',\n                'www': 'https://www.livecoin.net',\n                'doc': 'https://www.livecoin.net/api?lang=en',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'exchange/all/order_book',\n                        'exchange/last_trades',\n                        'exchange/maxbid_minask',\n                        'exchange/order_book',\n                        'exchange/restrictions',\n                        'exchange/ticker', // omit params to get all tickers at once\n                        'info/coinInfo',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'exchange/client_orders',\n                        'exchange/order',\n                        'exchange/trades',\n                        'exchange/commission',\n                        'exchange/commissionCommonInfo',\n                        'payment/balances',\n                        'payment/balance',\n                        'payment/get/address',\n                        'payment/history/size',\n                        'payment/history/transactions',\n                    ],\n                    'post': [\n                        'exchange/buylimit',\n                        'exchange/buymarket',\n                        'exchange/cancellimit',\n                        'exchange/selllimit',\n                        'exchange/sellmarket',\n                        'payment/out/capitalist',\n                        'payment/out/card',\n                        'payment/out/coin',\n                        'payment/out/okpay',\n                        'payment/out/payeer',\n                        'payment/out/perfectmoney',\n                        'payment/voucher/amount',\n                        'payment/voucher/make',\n                        'payment/voucher/redeem',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'maker': 0.18 / 100,\n                    'taker': 0.18 / 100,\n                },\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        return currency;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetExchangeTicker ();\n        let restrictions = await this.publicGetExchangeRestrictions ();\n        let restrictionsById = this.indexBy (restrictions['restrictions'], 'currencyPair');\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['symbol'];\n            let symbol = id;\n            let [ base, quote ] = symbol.split ('/');\n            let coinRestrictions = this.safeValue (restrictionsById, symbol);\n            let precision = {\n                'price': 5,\n                'amount': 8,\n                'cost': 8,\n            };\n            let limits = {\n                'amount': {\n                    'min': Math.pow (10, -precision['amount']),\n                    'max': Math.pow (10, precision['amount']),\n                },\n            };\n            if (coinRestrictions) {\n                precision['price'] = this.safeInteger (coinRestrictions, 'priceScale', 5);\n                limits['amount']['min'] = this.safeFloat (coinRestrictions, 'minLimitQuantity', limits['amount']['min']);\n            }\n            limits['price'] = {\n                'min': Math.pow (10, -precision['price']),\n                'max': Math.pow (10, precision['price']),\n            };\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'precision': precision,\n                'limits': limits,\n                'info': market,\n            }));\n        }\n        return result;\n    }\n\n    async fetchCurrencies (params = {}) {\n        let response = await this.publicGetInfoCoinInfo (params);\n        let currencies = response['info'];\n        let result = {};\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let id = currency['symbol'];\n            // todo: will need to rethink the fees\n            // to add support for multiple withdrawal/deposit methods and\n            // differentiated fees for each particular method\n            let code = this.commonCurrencyCode (id);\n            let precision = 8; // default precision, todo: fix \"magic constants\"\n            let active = (currency['walletStatus'] == 'normal');\n            result[code] = {\n                'id': id,\n                'code': code,\n                'info': currency,\n                'name': currency['name'],\n                'active': active,\n                'status': 'ok',\n                'fee': currency['withdrawFee'], // todo: redesign\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': currency['minOrderAmount'],\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': currency['minOrderAmount'],\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': currency['minWithdrawAmount'],\n                        'max': Math.pow (10, precision),\n                    },\n                    'deposit': {\n                        'min': currency['minDepositAmount'],\n                        'max': undefined,\n                    },\n                },\n            };\n        }\n        result = this.appendFiatCurrencies (result);\n        return result;\n    }\n\n    appendFiatCurrencies (result = []) {\n        let precision = 8;\n        let defaults = {\n            'info': undefined,\n            'active': true,\n            'status': 'ok',\n            'fee': undefined,\n            'precision': precision,\n            'limits': {\n                'withdraw': { 'min': undefined, 'max': undefined },\n                'deposit': { 'min': undefined, 'max': undefined },\n                'amount': { 'min': undefined, 'max': undefined },\n                'cost': { 'min': undefined, 'max': undefined },\n                'price': {\n                    'min': Math.pow (10, -precision),\n                    'max': Math.pow (10, precision),\n                },\n            },\n        };\n        let currencies = [\n            { 'id': 'USD', 'code': 'USD', 'name': 'US Dollar' },\n            { 'id': 'EUR', 'code': 'EUR', 'name': 'Euro' },\n            { 'id': 'RUR', 'code': 'RUR', 'name': 'Russian ruble' },\n        ];\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let code = currency['code'];\n            result[code] = this.extend (defaults, currency);\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privateGetPaymentBalances ();\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let account = undefined;\n            if (currency in result)\n                account = result[currency];\n            else\n                account = this.account ();\n            if (balance['type'] == 'total')\n                account['total'] = parseFloat (balance['value']);\n            if (balance['type'] == 'available')\n                account['free'] = parseFloat (balance['value']);\n            if (balance['type'] == 'trade')\n                account['used'] = parseFloat (balance['value']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchFees (params = {}) {\n        await this.loadMarkets ();\n        let commissionInfo = await this.privateGetExchangeCommissionCommonInfo ();\n        let commission = this.safeFloat (commissionInfo, 'commission');\n        return {\n            'info': commissionInfo,\n            'maker': commission,\n            'taker': commission,\n            'withdraw': 0.0,\n        };\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetExchangeOrderBook (this.extend ({\n            'currencyPair': this.marketId (symbol),\n            'groupByPrice': 'false',\n            'depth': 100,\n        }, params));\n        let timestamp = orderbook['timestamp'];\n        return this.parseOrderBook (orderbook, timestamp);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let vwap = parseFloat (ticker['vwap']);\n        let baseVolume = parseFloat (ticker['volume']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['best_bid']),\n            'ask': parseFloat (ticker['best_ask']),\n            'vwap': parseFloat (ticker['vwap']),\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetExchangeTicker (params);\n        let tickers = this.indexBy (response, 'symbol');\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetExchangeTicker (this.extend ({\n            'currencyPair': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['time'] * 1000;\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': trade['id'].toString (),\n            'order': undefined,\n            'type': undefined,\n            'side': trade['type'].toLowerCase (),\n            'price': trade['price'],\n            'amount': trade['quantity'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetExchangeLastTrades (this.extend ({\n            'currencyPair': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    parseOrder (order, market = undefined) {\n        let timestamp = this.safeInteger (order, 'lastModificationTime');\n        if (!timestamp)\n            timestamp = this.parse8601 (order['lastModificationTime']);\n        let trades = undefined;\n        if ('trades' in order)\n            // TODO currently not supported by livecoin\n            // trades = this.parseTrades (order['trades'], market, since, limit);\n            trades = undefined;\n        let status = undefined;\n        if (order['orderStatus'] == 'OPEN' || order['orderStatus'] == 'PARTIALLY_FILLED') {\n            status = 'open';\n        } else if (order['orderStatus'] == 'EXECUTED' || order['orderStatus'] == 'PARTIALLY_FILLED_AND_CANCELLED') {\n            status = 'closed';\n        } else {\n            status = 'canceled';\n        }\n        let symbol = order['currencyPair'];\n        let [ base, quote ] = symbol.split ('/');\n        let type = undefined;\n        let side = undefined;\n        if (order['type'].indexOf ('MARKET') >= 0) {\n            type = 'market';\n        } else {\n            type = 'limit';\n        }\n        if (order['type'].indexOf ('SELL') >= 0) {\n            side = 'sell';\n        } else {\n            side = 'buy';\n        }\n        let price = this.safeFloat (order, 'price', 0.0);\n        let cost = this.safeFloat (order, 'commissionByTrade', 0.0);\n        let remaining = this.safeFloat (order, 'remainingQuantity', 0.0);\n        let amount = this.safeFloat (order, 'quantity', remaining);\n        let filled = amount - remaining;\n        return {\n            'info': order,\n            'id': order['id'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': status,\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'trades': trades,\n            'fee': {\n                'cost': cost,\n                'currency': quote,\n            },\n        };\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        if (symbol)\n            market = this.market (symbol);\n        let pair = market ? market['id'] : undefined;\n        let request = {};\n        if (pair)\n            request['currencyPair'] = pair;\n        if (since)\n            request['issuedFrom'] = parseInt (since);\n        if (limit)\n            request['endRow'] = limit - 1;\n        let response = await this.privateGetExchangeClientOrders (this.extend (request, params));\n        let result = [];\n        let rawOrders = [];\n        if (response['data'])\n            rawOrders = response['data'];\n        for (let i = 0; i < rawOrders.length; i++) {\n            let order = rawOrders[i];\n            result.push (this.parseOrder (order, market));\n        }\n        return result;\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let result = await this.fetchOrders (symbol, since, limit, this.extend ({\n            'openClosed': 'OPEN',\n        }, params));\n        return result;\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let result = await this.fetchOrders (symbol, since, limit, this.extend ({\n            'openClosed': 'CLOSED',\n        }, params));\n        return result;\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let method = 'privatePostExchange' + this.capitalize (side) + type;\n        let market = this.market (symbol);\n        let order = {\n            'quantity': this.amountToPrecision (symbol, amount),\n            'currencyPair': market['id'],\n        };\n        if (type == 'limit')\n            order['price'] = this.priceToPrecision (symbol, price);\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['orderId'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' cancelOrder requires a symbol argument');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let currencyPair = market['id'];\n        let response = await this.privatePostExchangeCancellimit (this.extend ({\n            'orderId': id,\n            'currencyPair': currencyPair,\n        }, params));\n        let message = this.safeString (response, 'message', this.json (response));\n        if ('success' in response) {\n            if (!response['success']) {\n                throw new InvalidOrder (message);\n            } else if ('cancelled' in response) {\n                if (response['cancelled']) {\n                    return response;\n                } else {\n                    throw new OrderNotFound (message);\n                }\n            }\n        }\n        throw new ExchangeError (this.id + ' cancelOrder() failed: ' + this.json (response));\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        let request = {\n            'currency': currency,\n        };\n        let response = await this.privateGetPaymentGetAddress (this.extend (request, params));\n        let address = this.safeString (response, 'wallet');\n        return {\n            'currency': currency,\n            'address': address,\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + path;\n        let query = this.urlencode (this.keysort (params));\n        if (method == 'GET') {\n            if (Object.keys (params).length) {\n                url += '?' + query;\n            }\n        }\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            if (method == 'POST')\n                body = query;\n            let signature = this.hmac (this.encode (query), this.encode (this.secret), 'sha256');\n            headers = {\n                'Api-Key': this.apiKey,\n                'Sign': signature.toUpperCase (),\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code >= 300) {\n            if (body[0] == \"{\") {\n                let response = JSON.parse (body);\n                if ('errorCode' in response) {\n                    let error = response['errorCode'];\n                    if (error == 1) {\n                        throw new ExchangeError (this.id + ' ' + this.json (response));\n                    } else if (error == 2) {\n                        if ('errorMessage' in response) {\n                            if (response['errorMessage'] == 'User not found')\n                                throw new AuthenticationError (this.id + ' ' + response['errorMessage']);\n                        } else {\n                            throw new ExchangeError (this.id + ' ' + this.json (response));\n                        }\n                    } else if ((error == 10) || (error == 11) || (error == 12) || (error == 20) || (error == 30) || (error == 101) || (error == 102)) {\n                        throw new AuthenticationError (this.id + ' ' + this.json (response));\n                    } else if (error == 31) {\n                        throw new NotSupported (this.id + ' ' + this.json (response));\n                    } else if (error == 32) {\n                        throw new ExchangeError (this.id + ' ' + this.json (response));\n                    } else if (error == 100) {\n                        throw new ExchangeError (this.id + ': Invalid parameters ' + this.json (response));\n                    } else if (error == 103) {\n                        throw new InvalidOrder (this.id + ': Invalid currency ' + this.json (response));\n                    } else if (error == 104) {\n                        throw new InvalidOrder (this.id + ': Invalid amount ' + this.json (response));\n                    } else if (error == 105) {\n                        throw new InvalidOrder (this.id + ': Unable to block funds ' + this.json (response));\n                    } else if (error == 503) {\n                        throw new ExchangeNotAvailable (this.id + ': Exchange is not available ' + this.json (response));\n                    } else {\n                        throw new ExchangeError (this.id + ' ' + this.json (response));\n                    }\n                }\n            }\n            throw new ExchangeError (this.id + ' ' + body);\n        }\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('success' in response) {\n            if (!response['success']) {\n                throw new ExchangeError (this.id + ' error: ' + this.json (response));\n            }\n        }\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class luno extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'luno',\n            'name': 'luno',\n            'countries': [ 'GB', 'SG', 'ZA' ],\n            'rateLimit': 10000,\n            'version': '1',\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n                'fetchOrder': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766607-8c1a69d8-5ede-11e7-930c-540b5eb9be24.jpg',\n                'api': 'https://api.mybitx.com/api',\n                'www': 'https://www.luno.com',\n                'doc': [\n                    'https://www.luno.com/en/api',\n                    'https://npmjs.org/package/bitx',\n                    'https://github.com/bausmeier/node-bitx',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'orderbook',\n                        'ticker',\n                        'tickers',\n                        'trades',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'accounts/{id}/pending',\n                        'accounts/{id}/transactions',\n                        'balance',\n                        'fee_info',\n                        'funding_address',\n                        'listorders',\n                        'listtrades',\n                        'orders/{id}',\n                        'quotes/{id}',\n                        'withdrawals',\n                        'withdrawals/{id}',\n                    ],\n                    'post': [\n                        'accounts',\n                        'postorder',\n                        'marketorder',\n                        'stoporder',\n                        'funding_address',\n                        'withdrawals',\n                        'send',\n                        'quotes',\n                        'oauth2/grant',\n                    ],\n                    'put': [\n                        'quotes/{id}',\n                    ],\n                    'delete': [\n                        'quotes/{id}',\n                        'withdrawals/{id}',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetTickers ();\n        let result = [];\n        for (let p = 0; p < markets['tickers'].length; p++) {\n            let market = markets['tickers'][p];\n            let id = market['pair'];\n            let base = id.slice (0, 3);\n            let quote = id.slice (3, 6);\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetBalance ();\n        let balances = response['balance'];\n        let result = { 'info': response };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = this.commonCurrencyCode (balance['asset']);\n            let reserved = parseFloat (balance['reserved']);\n            let unconfirmed = parseFloat (balance['unconfirmed']);\n            let account = {\n                'free': 0.0,\n                'used': this.sum (reserved, unconfirmed),\n                'total': parseFloat (balance['balance']),\n            };\n            account['free'] = account['total'] - account['used'];\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetOrderbook (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        let timestamp = orderbook['timestamp'];\n        return this.parseOrderBook (orderbook, timestamp, 'bids', 'asks', 'price', 'volume');\n    }\n\n    parseOrder (order, market = undefined) {\n        let timestamp = order['creation_timestamp'];\n        let status = (order['state'] == 'PENDING') ? 'open' : 'closed';\n        let side = (order['type'] == 'ASK') ? 'sell' : 'buy';\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let price = this.safeFloat (order, 'limit_price');\n        let amount = this.safeFloat (order, 'limit_volume');\n        let quoteFee = this.safeFloat (order, 'fee_counter');\n        let baseFee = this.safeFloat (order, 'fee_base');\n        let fee = { 'currency': undefined };\n        if (quoteFee) {\n            fee['side'] = 'quote';\n            fee['cost'] = quoteFee;\n        } else {\n            fee['side'] = 'base';\n            fee['cost'] = baseFee;\n        }\n        return {\n            'id': order['order_id'],\n            'datetime': this.iso8601 (timestamp),\n            'timestamp': timestamp,\n            'status': status,\n            'symbol': symbol,\n            'type': undefined,\n            'side': side,\n            'price': price,\n            'amount': amount,\n            'filled': undefined,\n            'remaining': undefined,\n            'trades': undefined,\n            'fee': fee,\n            'info': order,\n        };\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetOrdersId (this.extend ({\n            'id': id,\n        }, params));\n        return this.parseOrder (response);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['timestamp'];\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last_trade']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['rolling_24_hour_volume']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetTickers (params);\n        let tickers = this.indexBy (response['tickers'], 'pair');\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetTicker (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let side = (trade['is_buy']) ? 'buy' : 'sell';\n        return {\n            'info': trade,\n            'id': undefined,\n            'order': undefined,\n            'timestamp': trade['timestamp'],\n            'datetime': this.iso8601 (trade['timestamp']),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': side,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['volume']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTrades (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTrades (response['trades'], market, since, limit);\n    }\n\n    async createOrder (market, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let method = 'privatePost';\n        let order = { 'pair': this.marketId (market) };\n        if (type == 'market') {\n            method += 'Marketorder';\n            order['type'] = side.toUpperCase ();\n            if (side == 'buy')\n                order['counter_volume'] = amount;\n            else\n                order['base_volume'] = amount;\n        } else {\n            method += 'Order';\n            order['volume'] = amount;\n            order['price'] = price;\n            if (side == 'buy')\n                order['type'] = 'BID';\n            else\n                order['type'] = 'ASK';\n        }\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['order_id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostStoporder ({ 'order_id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (Object.keys (query).length)\n            url += '?' + this.urlencode (query);\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            let auth = this.encode (this.apiKey + ':' + this.secret);\n            auth = this.stringToBase64 (auth);\n            headers = { 'Authorization': 'Basic ' + this.decode (auth) };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class lykke extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'lykke',\n            'name': 'Lykke',\n            'countries': 'CH',\n            'version': 'v1',\n            'rateLimit': 200,\n            'has': {\n                'CORS': false,\n                'fetchOHLCV': false,\n                'fetchTrades': false,\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/34487620-3139a7b0-efe6-11e7-90f5-e520cef74451.jpg',\n                'api': {\n                    'mobile': 'https://api.lykkex.com/api',\n                    'public': 'https://hft-api.lykke.com/api',\n                    'private': 'https://hft-api.lykke.com/api',\n                    'test': {\n                        'mobile': 'https://api.lykkex.com/api',\n                        'public': 'https://hft-service-dev.lykkex.net/api',\n                        'private': 'https://hft-service-dev.lykkex.net/api',\n                    },\n                },\n                'www': 'https://www.lykke.com',\n                'doc': [\n                    'https://hft-api.lykke.com/swagger/ui/',\n                    'https://www.lykke.com/lykke_api',\n                ],\n                'fees': 'https://www.lykke.com/trading-conditions',\n            },\n            'api': {\n                'mobile': {\n                    'get': [\n                        'AllAssetPairRates/{market}',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'AssetPairs',\n                        'AssetPairs/{id}',\n                        'IsAlive',\n                        'OrderBooks',\n                        'OrderBooks/{AssetPairId}',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'Orders',\n                        'Orders/{id}',\n                        'Wallets',\n                    ],\n                    'post': [\n                        'Orders/limit',\n                        'Orders/market',\n                        'Orders/{id}/Cancel',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'maker': 0.0010,\n                    'taker': 0.0019,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.001,\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privateGetWallets ();\n        let result = { 'info': balances };\n        for (let i = 0; i < balances.length; i++) {\n            let balance = balances[i];\n            let currency = balance['AssetId'];\n            let total = balance['Balance'];\n            let used = balance['Reserved'];\n            let free = total - used;\n            result[currency] = {\n                'free': free,\n                'used': used,\n                'total': total,\n            };\n        }\n        return this.parseBalance (result);\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostOrdersIdCancel ({ 'id': id });\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let query = {\n            'AssetPairId': market['id'],\n            'OrderAction': this.capitalize (side),\n            'Volume': amount,\n        };\n        if (type == 'market') {\n            query['Asset'] = (side == 'buy') ? market['base'] : market['quote'];\n        } else if (type == 'limit') {\n            query['Price'] = price;\n        }\n        let method = 'privatePostOrders' + this.capitalize (type);\n        let result = await this[method] (this.extend (query, params));\n        return {\n            'id': undefined,\n            'info': result,\n        };\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetAssetPairs ();\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            let id = market['Id'];\n            let base = market['BaseAssetId'];\n            let quote = market['QuotingAssetId'];\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = market['Name'];\n            let precision = {\n                'amount': market['Accuracy'],\n                'price': market['InvertedAccuracy'],\n            };\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': true,\n                'info': market,\n                'lot': Math.pow (10, -precision['amount']),\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision['amount']),\n                        'max': Math.pow (10, precision['amount']),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision['price']),\n                        'max': Math.pow (10, precision['price']),\n                    },\n                },\n            }));\n        }\n        return result;\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        ticker = ticker['Result'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': parseFloat (ticker['Rate']['Bid']),\n            'ask': parseFloat (ticker['Rate']['Ask']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': undefined,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.mobileGetAllAssetPairRatesMarket (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseOrderStatus (status) {\n        if (status == 'Pending') {\n            return 'open';\n        } else if (status == 'InOrderBook') {\n            return 'open';\n        } else if (status == 'Processing') {\n            return 'open';\n        } else if (status == 'Matched') {\n            return 'closed';\n        } else if (status == 'Cancelled') {\n            return 'canceled';\n        } else if (status == 'NotEnoughFunds') {\n            return 'NotEnoughFunds';\n        } else if (status == 'NoLiquidity') {\n            return 'NoLiquidity';\n        } else if (status == 'UnknownAsset') {\n            return 'UnknownAsset';\n        } else if (status == 'LeadToNegativeSpread') {\n            return 'LeadToNegativeSpread';\n        }\n        return status;\n    }\n\n    parseOrder (order, market = undefined) {\n        let status = this.parseOrderStatus (order['Status']);\n        let symbol = undefined;\n        if (!market) {\n            if ('AssetPairId' in order)\n                if (order['AssetPairId'] in this.markets_by_id)\n                    market = this.markets_by_id[order['AssetPairId']];\n        }\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = undefined;\n        if ('LastMatchTime' in order) {\n            timestamp = this.parse8601 (order['LastMatchTime']);\n        } else if ('Registered' in order) {\n            timestamp = this.parse8601 (order['Registered']);\n        } else if ('CreatedAt' in order) {\n            timestamp = this.parse8601 (order['CreatedAt']);\n        }\n        let price = this.safeFloat (order, 'Price');\n        let amount = this.safeFloat (order, 'Volume');\n        let remaining = this.safeFloat (order, 'RemainingVolume');\n        let filled = amount - remaining;\n        let cost = filled * price;\n        let result = {\n            'info': order,\n            'id': order['Id'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': undefined,\n            'side': undefined,\n            'price': price,\n            'cost': cost,\n            'average': undefined,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': undefined,\n        };\n        return result;\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        let response = await this.privateGetOrdersId (this.extend ({\n            'id': id,\n        }, params));\n        return this.parseOrder (response);\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let response = await this.privateGetOrders ();\n        return this.parseOrders (response, undefined, since, limit);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let response = await this.privateGetOrders (this.extend ({\n            'status': 'InOrderBook',\n        }, params));\n        return this.parseOrders (response, undefined, since, limit);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let response = await this.privateGetOrders (this.extend ({\n            'status': 'Matched',\n        }, params));\n        return this.parseOrders (response, undefined, since, limit);\n    }\n\n    async fetchOrderBook (symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetOrderBooksAssetPairId (this.extend ({\n            'AssetPairId': this.marketId (symbol),\n        }, params));\n        let orderbook = {\n            'timestamp': undefined,\n            'bids': [],\n            'asks': [],\n        };\n        let timestamp = undefined;\n        for (let i = 0; i < response.length; i++) {\n            let side = response[i];\n            if (side['IsBuy']) {\n                orderbook['bids'] = this.arrayConcat (orderbook['bids'], side['Prices']);\n            } else {\n                orderbook['asks'] = this.arrayConcat (orderbook['asks'], side['Prices']);\n            }\n            let timestamp = this.parse8601 (side['Timestamp']);\n            if (!orderbook['timestamp']) {\n                orderbook['timestamp'] = timestamp;\n            } else {\n                orderbook['timestamp'] = Math.max (orderbook['timestamp'], timestamp);\n            }\n        }\n        if (!timestamp)\n            timestamp = this.milliseconds ();\n        return this.parseOrderBook (orderbook, orderbook['timestamp'], 'bids', 'asks', 'Price', 'Volume');\n    }\n\n    parseBidAsk (bidask, priceKey = 0, amountKey = 1) {\n        let price = parseFloat (bidask[priceKey]);\n        let amount = parseFloat (bidask[amountKey]);\n        if (amount < 0)\n            amount = -amount;\n        return [ price, amount ];\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else if (api == 'private') {\n            if (method == 'GET')\n                if (Object.keys (query).length)\n                    url += '?' + this.urlencode (query);\n            this.checkRequiredCredentials ();\n            headers = {\n                'api-key': this.apiKey,\n                'Accept': 'application/json',\n                'Content-Type': 'application/json',\n            };\n            if (method == 'POST')\n                if (Object.keys (params).length)\n                    body = this.json (params);\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class mercado extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'mercado',\n            'name': 'Mercado Bitcoin',\n            'countries': 'BR', // Brazil\n            'rateLimit': 1000,\n            'version': 'v3',\n            'has': {\n                'CORS': true,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27837060-e7c58714-60ea-11e7-9192-f05e86adb83f.jpg',\n                'api': {\n                    'public': 'https://www.mercadobitcoin.net/api',\n                    'private': 'https://www.mercadobitcoin.net/tapi',\n                },\n                'www': 'https://www.mercadobitcoin.com.br',\n                'doc': [\n                    'https://www.mercadobitcoin.com.br/api-doc',\n                    'https://www.mercadobitcoin.com.br/trade-api',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        '{coin}/orderbook/', // last slash critical\n                        '{coin}/ticker/',\n                        '{coin}/trades/',\n                        '{coin}/trades/{from}/',\n                        '{coin}/trades/{from}/{to}',\n                        '{coin}/day-summary/{year}/{month}/{day}/',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'cancel_order',\n                        'get_account_info',\n                        'get_order',\n                        'get_withdrawal',\n                        'list_system_messages',\n                        'list_orders',\n                        'list_orderbook',\n                        'place_buy_order',\n                        'place_sell_order',\n                        'withdraw_coin',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/BRL': { 'id': 'BRLBTC', 'symbol': 'BTC/BRL', 'base': 'BTC', 'quote': 'BRL', 'suffix': 'Bitcoin' },\n                'LTC/BRL': { 'id': 'BRLLTC', 'symbol': 'LTC/BRL', 'base': 'LTC', 'quote': 'BRL', 'suffix': 'Litecoin' },\n                'BCH/BRL': { 'id': 'BRLBCH', 'symbol': 'BCH/BRL', 'base': 'BCH', 'quote': 'BRL', 'suffix': 'BCash' },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.3 / 100,\n                    'taker': 0.7 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let market = this.market (symbol);\n        let orderbook = await this.publicGetCoinOrderbook (this.extend ({\n            'coin': market['base'],\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetCoinTicker (this.extend ({\n            'coin': market['base'],\n        }, params));\n        let ticker = response['ticker'];\n        let timestamp = parseInt (ticker['date']) * 1000;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy']),\n            'ask': parseFloat (ticker['sell']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['vol']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['date'] * 1000;\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': trade['tid'].toString (),\n            'order': undefined,\n            'type': undefined,\n            'side': trade['type'],\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetCoinTrades (this.extend ({\n            'coin': market['base'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostGetAccountInfo ();\n        let balances = response['response_data']['balance'];\n        let result = { 'info': response };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let account = this.account ();\n            if (lowercase in balances) {\n                account['free'] = parseFloat (balances[lowercase]['available']);\n                account['total'] = parseFloat (balances[lowercase]['total']);\n                account['used'] = account['total'] - account['free'];\n            }\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type === 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let method = 'privatePostPlace' + this.capitalize (side) + 'Order';\n        let order = {\n            'coin_pair': this.marketId (symbol),\n            'quantity': amount,\n            'limit_price': price,\n        };\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['response_data']['order']['order_id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' cancelOrder() requires a symbol argument');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        return await this.privatePostCancelOrder (this.extend ({\n            'coin_pair': market['id'],\n            'order_id': id,\n        }, params));\n    }\n\n    parseOrder (order, market = undefined) {\n        let side = undefined;\n        if ('order_type' in order)\n            side = (order['order_type'] === 1) ? 'buy' : 'sell';\n        let status = order['status'];\n        let symbol = undefined;\n        if (!market) {\n            if ('coin_pair' in order)\n                if (order['coin_pair'] in this.markets_by_id)\n                    market = this.markets_by_id[order['coin_pair']];\n        }\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = undefined;\n        if ('created_timestamp' in order)\n            timestamp = parseInt (order['created_timestamp']) * 1000;\n        if ('updated_timestamp' in order)\n            timestamp = parseInt (order['updated_timestamp']) * 1000;\n        let fee = {\n            'cost': parseFloat (order['fee']),\n            'currency': market['quote'],\n        };\n        let price = this.safeFloat (order, 'limit_price');\n        // price = this.safeFloat (order, 'executed_price_avg', price);\n        let average = this.safeFloat (order, 'executed_price_avg');\n        let amount = this.safeFloat (order, 'quantity');\n        let filled = this.safeFloat (order, 'executed_quantity');\n        let remaining = amount - filled;\n        let cost = amount * average;\n        let result = {\n            'info': order,\n            'id': order['order_id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': 'limit',\n            'side': side,\n            'price': price,\n            'cost': cost,\n            'average': average,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': fee,\n        };\n        return result;\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' cancelOrder() requires a symbol argument');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = undefined;\n        response = await this.privatePostGetOrder (this.extend ({\n            'coin_pair': market['id'],\n            'order_id': parseInt (id),\n        }, params));\n        return this.parseOrder (response['response_data']['order']);\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            'coin': currency,\n            'quantity': amount.toFixed (10),\n            'address': address,\n        };\n        if (currency === 'BRL') {\n            let account_ref = ('account_ref' in params);\n            if (!account_ref)\n                throw new ExchangeError (this.id + ' requires account_ref parameter to withdraw ' + currency);\n        } else if (currency !== 'LTC') {\n            let tx_fee = ('tx_fee' in params);\n            if (!tx_fee)\n                throw new ExchangeError (this.id + ' requires tx_fee parameter to withdraw ' + currency);\n        }\n        let response = await this.privatePostWithdrawCoin (this.extend (request, params));\n        return {\n            'info': response,\n            'id': response['response_data']['withdrawal']['id'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api] + '/';\n        if (api === 'public') {\n            url += this.implodeParams (path, params);\n        } else {\n            this.checkRequiredCredentials ();\n            url += this.version + '/';\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({\n                'tapi_method': path,\n                'tapi_nonce': nonce,\n            }, params));\n            let auth = '/tapi/' + this.version + '/' + '?' + body;\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'TAPI-ID': this.apiKey,\n                'TAPI-MAC': this.hmac (this.encode (auth), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error_message' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class mixcoins extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'mixcoins',\n            'name': 'MixCoins',\n            'countries': [ 'GB', 'HK' ],\n            'rateLimit': 1500,\n            'version': 'v1',\n            'has': {\n                'CORS': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30237212-ed29303c-9535-11e7-8af8-fcd381cfa20c.jpg',\n                'api': 'https://mixcoins.com/api',\n                'www': 'https://mixcoins.com',\n                'doc': 'https://mixcoins.com/help/api/',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'ticker',\n                        'trades',\n                        'depth',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'cancel',\n                        'info',\n                        'orders',\n                        'order',\n                        'transactions',\n                        'trade',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/USD': { 'id': 'btc_usd', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD', 'maker': 0.0015, 'taker': 0.0025 },\n                'ETH/BTC': { 'id': 'eth_btc', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC', 'maker': 0.001, 'taker': 0.0015 },\n                'BCH/BTC': { 'id': 'bch_btc', 'symbol': 'BCH/BTC', 'base': 'BCH', 'quote': 'BTC', 'maker': 0.001, 'taker': 0.0015 },\n                'LSK/BTC': { 'id': 'lsk_btc', 'symbol': 'LSK/BTC', 'base': 'LSK', 'quote': 'BTC', 'maker': 0.0015, 'taker': 0.0025 },\n                'BCH/USD': { 'id': 'bch_usd', 'symbol': 'BCH/USD', 'base': 'BCH', 'quote': 'USD', 'maker': 0.001, 'taker': 0.0015 },\n                'ETH/USD': { 'id': 'eth_usd', 'symbol': 'ETH/USD', 'base': 'ETH', 'quote': 'USD', 'maker': 0.001, 'taker': 0.0015 },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let response = await this.privatePostInfo ();\n        let balance = response['result']['wallet'];\n        let result = { 'info': balance };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let account = this.account ();\n            if (lowercase in balance) {\n                account['free'] = parseFloat (balance[lowercase]['avail']);\n                account['used'] = parseFloat (balance[lowercase]['lock']);\n                account['total'] = this.sum (account['free'], account['used']);\n            }\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let response = await this.publicGetDepth (this.extend ({\n            'market': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (response['result']);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let response = await this.publicGetTicker (this.extend ({\n            'market': this.marketId (symbol),\n        }, params));\n        let ticker = response['result'];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy']),\n            'ask': parseFloat (ticker['sell']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['vol']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        return {\n            'id': trade['id'].toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': undefined,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetTrades (this.extend ({\n            'market': market['id'],\n        }, params));\n        return this.parseTrades (response['result'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let order = {\n            'market': this.marketId (symbol),\n            'op': side,\n            'amount': amount,\n        };\n        if (type == 'market') {\n            order['order_type'] = 1;\n            order['price'] = price;\n        } else {\n            order['order_type'] = 0;\n        }\n        let response = await this.privatePostTrade (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['result']['id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancel ({ 'id': id });\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + path;\n        if (api == 'public') {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({\n                'nonce': nonce,\n            }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Key': this.apiKey,\n                'Sign': this.hmac (this.encode (body), this.secret, 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('status' in response)\n            if (response['status'] == 200)\n                return response;\n        throw new ExchangeError (this.id + ' ' + this.json (response));\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class nova extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'nova',\n            'name': 'Novaexchange',\n            'countries': 'TZ', // Tanzania\n            'rateLimit': 2000,\n            'version': 'v2',\n            'has': {\n                'CORS': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30518571-78ca0bca-9b8a-11e7-8840-64b83a4a94b2.jpg',\n                'api': 'https://novaexchange.com/remote',\n                'www': 'https://novaexchange.com',\n                'doc': 'https://novaexchange.com/remote/faq',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'markets/',\n                        'markets/{basecurrency}/',\n                        'market/info/{pair}/',\n                        'market/orderhistory/{pair}/',\n                        'market/openorders/{pair}/buy/',\n                        'market/openorders/{pair}/sell/',\n                        'market/openorders/{pair}/both/',\n                        'market/openorders/{pair}/{ordertype}/',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'getbalances/',\n                        'getbalance/{currency}/',\n                        'getdeposits/',\n                        'getwithdrawals/',\n                        'getnewdepositaddress/{currency}/',\n                        'getdepositaddress/{currency}/',\n                        'myopenorders/',\n                        'myopenorders_market/{pair}/',\n                        'cancelorder/{orderid}/',\n                        'withdraw/{currency}/',\n                        'trade/{pair}/',\n                        'tradehistory/',\n                        'getdeposithistory/',\n                        'getwithdrawalhistory/',\n                        'walletstatus/',\n                        'walletstatus/{currency}/',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.publicGetMarkets ();\n        let markets = response['markets'];\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let market = markets[i];\n            if (!market['disabled']) {\n                let id = market['marketname'];\n                let [ quote, base ] = id.split ('_');\n                let symbol = base + '/' + quote;\n                result.push ({\n                    'id': id,\n                    'symbol': symbol,\n                    'base': base,\n                    'quote': quote,\n                    'info': market,\n                });\n            }\n        }\n        return result;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetMarketOpenordersPairBoth (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'buyorders', 'sellorders', 'price', 'amount');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetMarketInfoPair (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        let ticker = response['markets'][0];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high24h']),\n            'low': parseFloat (ticker['low24h']),\n            'bid': this.safeFloat (ticker, 'bid'),\n            'ask': this.safeFloat (ticker, 'ask'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last_price']),\n            'change': parseFloat (ticker['change24h']),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': parseFloat (ticker['volume24h']),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['unix_t_datestamp'] * 1000;\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': undefined,\n            'order': undefined,\n            'type': undefined,\n            'side': trade['tradetype'].toLowerCase (),\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetMarketOrderhistoryPair (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTrades (response['items'], market, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetbalances ();\n        let balances = response['balances'];\n        let result = { 'info': response };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let lockbox = parseFloat (balance['amount_lockbox']);\n            let trades = parseFloat (balance['amount_trades']);\n            let account = {\n                'free': parseFloat (balance['amount']),\n                'used': this.sum (lockbox, trades),\n                'total': parseFloat (balance['amount_total']),\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type == 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        await this.loadMarkets ();\n        amount = amount.toString ();\n        price = price.toString ();\n        let market = this.market (symbol);\n        let order = {\n            'tradetype': side.toUpperCase (),\n            'tradeamount': amount,\n            'tradeprice': price,\n            'tradebase': 1,\n            'pair': market['id'],\n        };\n        let response = await this.privatePostTradePair (this.extend (order, params));\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelorder (this.extend ({\n            'orderid': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/';\n        if (api == 'private')\n            url += api + '/';\n        url += this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            url += '?' + this.urlencode ({ 'nonce': nonce });\n            let signature = this.hmac (this.encode (url), this.encode (this.secret), 'sha512', 'base64');\n            body = this.urlencode (this.extend ({\n                'apikey': this.apiKey,\n                'signature': signature,\n            }, query));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('status' in response)\n            if (response['status'] != 'success')\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst okcoinusd = require ('./okcoinusd.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class okcoincny extends okcoinusd {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'okcoincny',\n            'name': 'OKCoin CNY',\n            'countries': 'CN',\n            'has': {\n                'CORS': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766792-8be9157a-5ee5-11e7-926c-6d69b8d3378d.jpg',\n                'api': {\n                    'web': 'https://www.okcoin.cn',\n                    'public': 'https://www.okcoin.cn/pai',\n                    'private': 'https://www.okcoin.cn/api',\n                },\n                'www': 'https://www.okcoin.cn',\n                'doc': 'https://www.okcoin.cn/rest_getStarted.html',\n            },\n            'markets': {\n                'BTC/CNY': { 'id': 'btc_cny', 'symbol': 'BTC/CNY', 'base': 'BTC', 'quote': 'CNY', 'type': 'spot', 'spot': true, 'future': false },\n                'LTC/CNY': { 'id': 'ltc_cny', 'symbol': 'LTC/CNY', 'base': 'LTC', 'quote': 'CNY', 'type': 'spot', 'spot': true, 'future': false },\n                'ETH/CNY': { 'id': 'eth_cny', 'symbol': 'ETH/CNY', 'base': 'ETH', 'quote': 'CNY', 'type': 'spot', 'spot': true, 'future': false },\n                'ETC/CNY': { 'id': 'etc_cny', 'symbol': 'ETC/CNY', 'base': 'ETC', 'quote': 'CNY', 'type': 'spot', 'spot': true, 'future': false },\n                'BCH/CNY': { 'id': 'bcc_cny', 'symbol': 'BCH/CNY', 'base': 'BCH', 'quote': 'CNY', 'type': 'spot', 'spot': true, 'future': false },\n            },\n        });\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, InsufficientFunds, InvalidOrder, OrderNotFound, AuthenticationError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class okcoinusd extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'okcoinusd',\n            'name': 'OKCoin USD',\n            'countries': [ 'CN', 'US' ],\n            'version': 'v1',\n            'rateLimit': 1000, // up to 3000 requests per 5 minutes ≈ 600 requests per minute ≈ 10 requests per second ≈ 100 ms\n            'has': {\n                'CORS': false,\n                'fetchOHLCV': true,\n                'fetchOrder': true,\n                'fetchOrders': false,\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': true,\n                'withdraw': true,\n                'futureMarkets': false,\n            },\n            'extension': '.do', // appended to endpoint URL\n            'timeframes': {\n                '1m': '1min',\n                '3m': '3min',\n                '5m': '5min',\n                '15m': '15min',\n                '30m': '30min',\n                '1h': '1hour',\n                '2h': '2hour',\n                '4h': '4hour',\n                '6h': '6hour',\n                '12h': '12hour',\n                '1d': '1day',\n                '3d': '3day',\n                '1w': '1week',\n            },\n            'api': {\n                'web': {\n                    'get': [\n                        'markets/currencies',\n                        'markets/products',\n                    ],\n                },\n                'public': {\n                    'get': [\n                        'depth',\n                        'exchange_rate',\n                        'future_depth',\n                        'future_estimated_price',\n                        'future_hold_amount',\n                        'future_index',\n                        'future_kline',\n                        'future_price_limit',\n                        'future_ticker',\n                        'future_trades',\n                        'kline',\n                        'otcs',\n                        'ticker',\n                        'trades',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'account_records',\n                        'batch_trade',\n                        'borrow_money',\n                        'borrow_order_info',\n                        'borrows_info',\n                        'cancel_borrow',\n                        'cancel_order',\n                        'cancel_otc_order',\n                        'cancel_withdraw',\n                        'future_batch_trade',\n                        'future_cancel',\n                        'future_devolve',\n                        'future_explosive',\n                        'future_order_info',\n                        'future_orders_info',\n                        'future_position',\n                        'future_position_4fix',\n                        'future_trade',\n                        'future_trades_history',\n                        'future_userinfo',\n                        'future_userinfo_4fix',\n                        'lend_depth',\n                        'order_fee',\n                        'order_history',\n                        'order_info',\n                        'orders_info',\n                        'otc_order_history',\n                        'otc_order_info',\n                        'repayment',\n                        'submit_otc_order',\n                        'trade',\n                        'trade_history',\n                        'trade_otc_order',\n                        'withdraw',\n                        'withdraw_info',\n                        'unrepayments_info',\n                        'userinfo',\n                    ],\n                },\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766791-89ffb502-5ee5-11e7-8a5b-c5950b68ac65.jpg',\n                'api': {\n                    'web': 'https://www.okcoin.com/v2',\n                    'public': 'https://www.okcoin.com/api',\n                    'private': 'https://www.okcoin.com/api',\n                },\n                'www': 'https://www.okcoin.com',\n                'doc': [\n                    'https://www.okcoin.com/rest_getStarted.html',\n                    'https://www.npmjs.com/package/okcoin.com',\n                ],\n            },\n            'fees': {\n                'trading': {\n                    'taker': 0.002,\n                    'maker': 0.002,\n                },\n            },\n            'exceptions': {\n                '1009': OrderNotFound,\n                '1013': InvalidOrder, // no order type\n                '1027': InvalidOrder, // createLimitBuyOrder(symbol, 0, 0): Incorrect parameter may exceeded limits\n                '1002': InsufficientFunds, // The transaction amount exceed the balance\n                '10000': ExchangeError, // createLimitBuyOrder(symbol, undefined, undefined)\n                '10005': AuthenticationError, // bad apiKey\n                '10008': ExchangeError, // Illegal URL parameter\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let response = await this.webGetMarketsProducts ();\n        let markets = response['data'];\n        let result = [];\n        for (let i = 0; i < markets.length; i++) {\n            let id = markets[i]['symbol'];\n            let uppercase = id.toUpperCase ();\n            let [ base, quote ] = uppercase.split ('_');\n            let symbol = base + '/' + quote;\n            let precision = {\n                'amount': markets[i]['maxSizeDigit'],\n                'price': markets[i]['maxPriceDigit'],\n            };\n            let lot = Math.pow (10, -precision['amount']);\n            let minAmount = markets[i]['minTradeSize'];\n            let minPrice = Math.pow (10, -precision['price']);\n            let market = this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': markets[i],\n                'type': 'spot',\n                'spot': true,\n                'future': false,\n                'lot': lot,\n                'active': true,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': minAmount,\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': minPrice,\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': minAmount * minPrice,\n                        'max': undefined,\n                    },\n                },\n            });\n            result.push (market);\n            if ((this.hasFutureMarkets) && (market['quote'] === 'USDT')) {\n                result.push (this.extend (market, {\n                    'quote': 'USD',\n                    'symbol': market['base'] + '/USD',\n                    'id': market['id'].replace ('usdt', 'usd'),\n                    'type': 'future',\n                    'spot': false,\n                    'future': true,\n                }));\n            }\n        }\n        return result;\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'publicGet';\n        let request = {\n            'symbol': market['id'],\n        };\n        if (market['future']) {\n            method += 'Future';\n            request['contract_type'] = 'this_week'; // next_week, quarter\n        }\n        method += 'Depth';\n        let orderbook = await this[method] (this.extend (request, params));\n        let timestamp = this.milliseconds ();\n        return {\n            'bids': orderbook['bids'],\n            'asks': this.sortBy (orderbook['asks'], 0),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n        };\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['timestamp'];\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy']),\n            'ask': parseFloat (ticker['sell']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['vol']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'publicGet';\n        let request = {\n            'symbol': market['id'],\n        };\n        if (market['future']) {\n            method += 'Future';\n            request['contract_type'] = 'this_week'; // next_week, quarter\n        }\n        method += 'Ticker';\n        let response = await this[method] (this.extend (request, params));\n        let timestamp = parseInt (response['date']) * 1000;\n        let ticker = this.extend (response['ticker'], { 'timestamp': timestamp });\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'info': trade,\n            'timestamp': trade['date_ms'],\n            'datetime': this.iso8601 (trade['date_ms']),\n            'symbol': symbol,\n            'id': trade['tid'].toString (),\n            'order': undefined,\n            'type': undefined,\n            'side': trade['type'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'publicGet';\n        let request = {\n            'symbol': market['id'],\n        };\n        if (market['future']) {\n            method += 'Future';\n            request['contract_type'] = 'this_week'; // next_week, quarter\n        }\n        method += 'Trades';\n        let response = await this[method] (this.extend (request, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = 1440, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'publicGet';\n        let request = {\n            'symbol': market['id'],\n            'type': this.timeframes[timeframe],\n        };\n        if (market['future']) {\n            method += 'Future';\n            request['contract_type'] = 'this_week'; // next_week, quarter\n        }\n        method += 'Kline';\n        if (limit)\n            request['size'] = parseInt (limit);\n        if (since) {\n            request['since'] = since;\n        } else {\n            request['since'] = this.milliseconds () - 86400000; // last 24 hours\n        }\n        let response = await this[method] (this.extend (request, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostUserinfo ();\n        let balances = response['info']['funds'];\n        let result = { 'info': response };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let account = this.account ();\n            account['free'] = this.safeFloat (balances['free'], lowercase, 0.0);\n            account['used'] = this.safeFloat (balances['freezed'], lowercase, 0.0);\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'privatePost';\n        let order = {\n            'symbol': market['id'],\n            'type': side,\n        };\n        if (market['future']) {\n            method += 'Future';\n            order = this.extend (order, {\n                'contract_type': 'this_week', // next_week, quarter\n                'match_price': 0, // match best counter party price? 0 or 1, ignores price if 1\n                'lever_rate': 10, // leverage rate value: 10 or 20 (10 by default)\n                'price': price,\n                'amount': amount,\n            });\n        } else {\n            if (type === 'limit') {\n                order['price'] = price;\n                order['amount'] = amount;\n            } else {\n                order['type'] += '_market';\n                if (side === 'buy') {\n                    order['price'] = this.safeFloat (params, 'cost');\n                    if (!order['price'])\n                        throw new ExchangeError (this.id + ' market buy orders require an additional cost parameter, cost = price * amount');\n                } else {\n                    order['amount'] = amount;\n                }\n            }\n        }\n        params = this.omit (params, 'cost');\n        method += 'Trade';\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['order_id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' cancelOrder() requires a symbol argument');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'symbol': market['id'],\n            'order_id': id,\n        };\n        let method = 'privatePost';\n        if (market['future']) {\n            method += 'FutureCancel';\n            request['contract_type'] = 'this_week'; // next_week, quarter\n        } else {\n            method += 'CancelOrder';\n        }\n        let response = await this[method] (this.extend (request, params));\n        return response;\n    }\n\n    parseOrderStatus (status) {\n        if (status === -1)\n            return 'canceled';\n        if (status === 0)\n            return 'open';\n        if (status === 1)\n            return 'partial';\n        if (status === 2)\n            return 'closed';\n        if (status === 4)\n            return 'canceled';\n        return status;\n    }\n\n    parseOrder (order, market = undefined) {\n        let side = undefined;\n        let type = undefined;\n        if ('type' in order) {\n            if ((order['type'] === 'buy') || (order['type'] === 'sell')) {\n                side = order['type'];\n                type = 'limit';\n            } else {\n                side = (order['type'] === 'buy_market') ? 'buy' : 'sell';\n                type = 'market';\n            }\n        }\n        let status = this.parseOrderStatus (order['status']);\n        let symbol = undefined;\n        if (!market) {\n            if ('symbol' in order)\n                if (order['symbol'] in this.markets_by_id)\n                    market = this.markets_by_id[order['symbol']];\n        }\n        if (market)\n            symbol = market['symbol'];\n        let timestamp = undefined;\n        let createDateField = this.getCreateDateField ();\n        if (createDateField in order)\n            timestamp = order[createDateField];\n        let amount = order['amount'];\n        let filled = order['deal_amount'];\n        let remaining = amount - filled;\n        let average = order['avg_price'];\n        let cost = average * filled;\n        let result = {\n            'info': order,\n            'id': order['order_id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'type': type,\n            'side': side,\n            'price': order['price'],\n            'average': average,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'status': status,\n            'fee': undefined,\n        };\n        return result;\n    }\n\n    getCreateDateField () {\n        // needed for derived exchanges\n        // allcoin typo create_data instead of create_date\n        return 'create_date';\n    }\n\n    getOrdersField () {\n        // needed for derived exchanges\n        // allcoin typo order instead of orders (expected based on their API docs)\n        return 'orders';\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrder requires a symbol parameter');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'privatePost';\n        let request = {\n            'order_id': id,\n            'symbol': market['id'],\n            // 'status': 0, // 0 for unfilled orders, 1 for filled orders\n            // 'current_page': 1, // current page number\n            // 'page_length': 200, // number of orders returned per page, maximum 200\n        };\n        if (market['future']) {\n            method += 'Future';\n            request['contract_type'] = 'this_week'; // next_week, quarter\n        }\n        method += 'OrderInfo';\n        let response = await this[method] (this.extend (request, params));\n        let ordersField = this.getOrdersField ();\n        if (response[ordersField].length > 0)\n            return this.parseOrder (response[ordersField][0]);\n        throw new OrderNotFound (this.id + ' order ' + id + ' not found');\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        if (!symbol)\n            throw new ExchangeError (this.id + ' fetchOrders requires a symbol parameter');\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'privatePost';\n        let request = {\n            'symbol': market['id'],\n        };\n        let order_id_in_params = ('order_id' in params);\n        if (market['future']) {\n            method += 'FutureOrdersInfo';\n            request['contract_type'] = 'this_week'; // next_week, quarter\n            if (!order_id_in_params)\n                throw new ExchangeError (this.id + ' fetchOrders() requires order_id param for futures market ' + symbol + ' (a string of one or more order ids, comma-separated)');\n        } else {\n            let status = undefined;\n            if ('type' in params) {\n                status = params['type'];\n            } else if ('status' in params) {\n                status = params['status'];\n            } else {\n                let name = order_id_in_params ? 'type' : 'status';\n                throw new ExchangeError (this.id + ' fetchOrders() requires ' + name + ' param for spot market ' + symbol + ' (0 - for unfilled orders, 1 - for filled/canceled orders)');\n            }\n            if (order_id_in_params) {\n                method += 'OrdersInfo';\n                request = this.extend (request, {\n                    'type': status,\n                });\n            } else {\n                method += 'OrderHistory';\n                request = this.extend (request, {\n                    'status': status,\n                    'current_page': 1, // current page number\n                    'page_length': 200, // number of orders returned per page, maximum 200\n                });\n            }\n            params = this.omit (params, [ 'type', 'status' ]);\n        }\n        let response = await this[method] (this.extend (request, params));\n        let ordersField = this.getOrdersField ();\n        return this.parseOrders (response[ordersField], market, since, limit);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let open = 0; // 0 for unfilled orders, 1 for filled orders\n        return await this.fetchOrders (symbol, undefined, undefined, this.extend ({\n            'status': open,\n        }, params));\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let closed = 1; // 0 for unfilled orders, 1 for filled orders\n        let orders = await this.fetchOrders (symbol, undefined, undefined, this.extend ({\n            'status': closed,\n        }, params));\n        return this.filterBy (orders, 'status', 'closed');\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let lowercase = currency.toLowerCase () + '_usd';\n        // if (amount < 0.01)\n        //     throw new ExchangeError (this.id + ' withdraw() requires amount > 0.01');\n        let request = {\n            'symbol': lowercase,\n            'withdraw_address': address,\n            'withdraw_amount': amount,\n            'target': 'address', // or okcn, okcom, okex\n        };\n        let query = params;\n        if ('chargefee' in query) {\n            request['chargefee'] = query['chargefee'];\n            query = this.omit (query, 'chargefee');\n        } else {\n            throw new ExchangeError (this.id + ' withdraw() requires a `chargefee` parameter');\n        }\n        let password = undefined;\n        if (this.password) {\n            request['trade_pwd'] = this.password;\n            password = this.password;\n        } else if ('password' in query) {\n            request['trade_pwd'] = query['password'];\n            query = this.omit (query, 'password');\n        } else if ('trade_pwd' in query) {\n            request['trade_pwd'] = query['trade_pwd'];\n            query = this.omit (query, 'trade_pwd');\n        }\n        if (!password)\n            throw new ExchangeError (this.id + ' withdraw() requires this.password set on the exchange instance or a password / trade_pwd parameter');\n        let response = await this.privatePostWithdraw (this.extend (request, query));\n        return {\n            'info': response,\n            'id': this.safeString (response, 'withdraw_id'),\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = '/';\n        if (api !== 'web')\n            url += this.version + '/';\n        url += path + this.extension;\n        if (api === 'private') {\n            this.checkRequiredCredentials ();\n            let query = this.keysort (this.extend ({\n                'api_key': this.apiKey,\n            }, params));\n            // secret key must be at the end of query\n            let queryString = this.rawencode (query) + '&secret_key=' + this.secret;\n            query['sign'] = this.hash (this.encode (queryString)).toUpperCase ();\n            body = this.urlencode (query);\n            headers = { 'Content-Type': 'application/x-www-form-urlencoded' };\n        } else {\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        }\n        url = this.urls['api'][api] + url;\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (body.length < 2)\n            return; // fallback to default error handler\n        if (body[0] === '{') {\n            let response = JSON.parse (body);\n            if ('error_code' in response) {\n                let error = this.safeString (response, 'error_code');\n                let message = this.id + ' ' + this.json (response);\n                if (error in this.exceptions) {\n                    let ExceptionClass = this.exceptions[error];\n                    throw new ExceptionClass (message);\n                } else {\n                    throw new ExchangeError (message);\n                }\n            }\n            if ('result' in response)\n                if (!response['result'])\n                    throw new ExchangeError (this.id + ' ' + this.json (response));\n        }\n    }\n};\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst okcoinusd = require ('./okcoinusd.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class okex extends okcoinusd {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'okex',\n            'name': 'OKEX',\n            'countries': [ 'CN', 'US' ],\n            'has': {\n                'CORS': false,\n                'futureMarkets': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/32552768-0d6dd3c6-c4a6-11e7-90f8-c043b64756a7.jpg',\n                'api': {\n                    'web': 'https://www.okex.com/v2',\n                    'public': 'https://www.okex.com/api',\n                    'private': 'https://www.okex.com/api',\n                },\n                'www': 'https://www.okex.com',\n                'doc': 'https://www.okex.com/rest_getStarted.html',\n                'fees': 'https://www.okex.com/fees.html',\n            },\n        });\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class paymium extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'paymium',\n            'name': 'Paymium',\n            'countries': [ 'FR', 'EU' ],\n            'rateLimit': 2000,\n            'version': 'v1',\n            'has': {\n                'CORS': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27790564-a945a9d4-5ff9-11e7-9d2d-b635763f2f24.jpg',\n                'api': 'https://paymium.com/api',\n                'www': 'https://www.paymium.com',\n                'doc': [\n                    'https://github.com/Paymium/api-documentation',\n                    'https://www.paymium.com/page/developers',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'countries',\n                        'data/{id}/ticker',\n                        'data/{id}/trades',\n                        'data/{id}/depth',\n                        'bitcoin_charts/{id}/trades',\n                        'bitcoin_charts/{id}/depth',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'merchant/get_payment/{UUID}',\n                        'user',\n                        'user/addresses',\n                        'user/addresses/{btc_address}',\n                        'user/orders',\n                        'user/orders/{UUID}',\n                        'user/price_alerts',\n                    ],\n                    'post': [\n                        'user/orders',\n                        'user/addresses',\n                        'user/payment_requests',\n                        'user/price_alerts',\n                        'merchant/create_payment',\n                    ],\n                    'delete': [\n                        'user/orders/{UUID}/cancel',\n                        'user/price_alerts/{id}',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/EUR': { 'id': 'eur', 'symbol': 'BTC/EUR', 'base': 'BTC', 'quote': 'EUR' },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.0059,\n                    'taker': 0.0059,\n                },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let balances = await this.privateGetUser ();\n        let result = { 'info': balances };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let account = this.account ();\n            let balance = 'balance_' + lowercase;\n            let locked = 'locked_' + lowercase;\n            if (balance in balances)\n                account['free'] = balances[balance];\n            if (locked in balances)\n                account['used'] = balances[locked];\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetDataIdDepth (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        let result = this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'price', 'amount');\n        result['bids'] = this.sortBy (result['bids'], 0, true);\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetDataIdTicker (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        let timestamp = ticker['at'] * 1000;\n        let vwap = parseFloat (ticker['vwap']);\n        let baseVolume = parseFloat (ticker['volume']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'bid'),\n            'ask': this.safeFloat (ticker, 'ask'),\n            'vwap': vwap,\n            'open': this.safeFloat (ticker, 'open'),\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'price'),\n            'change': undefined,\n            'percentage': this.safeFloat (ticker, 'variation'),\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['created_at_int']) * 1000;\n        let volume = 'traded_' + market['base'].toLowerCase ();\n        return {\n            'info': trade,\n            'id': trade['uuid'],\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['side'],\n            'price': trade['price'],\n            'amount': trade[volume],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetDataIdTrades (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (market, type, side, amount, price = undefined, params = {}) {\n        let order = {\n            'type': this.capitalize (type) + 'Order',\n            'currency': this.marketId (market),\n            'direction': side,\n            'amount': amount,\n        };\n        if (type == 'market')\n            order['price'] = price;\n        let response = await this.privatePostUserOrders (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['uuid'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelOrder (this.extend ({\n            'orderNumber': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            body = this.json (params);\n            let nonce = this.nonce ().toString ();\n            let auth = nonce + url + body;\n            headers = {\n                'Api-Key': this.apiKey,\n                'Api-Signature': this.hmac (this.encode (auth), this.secret),\n                'Api-Nonce': nonce,\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('errors' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeNotAvailable, ExchangeError, InsufficientFunds, OrderNotFound, OrderNotCached, InvalidOrder } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class poloniex extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'poloniex',\n            'name': 'Poloniex',\n            'countries': 'US',\n            'rateLimit': 1000, // up to 6 calls per second\n            'has': {\n                'CORS': true,\n                'fetchOHLCV': true,\n                'fetchMyTrades': true,\n                'fetchOrder': 'emulated',\n                'fetchOrders': 'emulated',\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': 'emulated',\n                'fetchTickers': true,\n                'fetchCurrencies': true,\n                'withdraw': true,\n            },\n            'timeframes': {\n                '5m': 300,\n                '15m': 900,\n                '30m': 1800,\n                '2h': 7200,\n                '4h': 14400,\n                '1d': 86400,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766817-e9456312-5ee6-11e7-9b3c-b628ca5626a5.jpg',\n                'api': {\n                    'public': 'https://poloniex.com/public',\n                    'private': 'https://poloniex.com/tradingApi',\n                },\n                'www': 'https://poloniex.com',\n                'doc': [\n                    'https://poloniex.com/support/api/',\n                    'http://pastebin.com/dMX7mZE0',\n                ],\n                'fees': 'https://poloniex.com/fees',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'return24hVolume',\n                        'returnChartData',\n                        'returnCurrencies',\n                        'returnLoanOrders',\n                        'returnOrderBook',\n                        'returnTicker',\n                        'returnTradeHistory',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'buy',\n                        'cancelLoanOffer',\n                        'cancelOrder',\n                        'closeMarginPosition',\n                        'createLoanOffer',\n                        'generateNewAddress',\n                        'getMarginPosition',\n                        'marginBuy',\n                        'marginSell',\n                        'moveOrder',\n                        'returnActiveLoans',\n                        'returnAvailableAccountBalances',\n                        'returnBalances',\n                        'returnCompleteBalances',\n                        'returnDepositAddresses',\n                        'returnDepositsWithdrawals',\n                        'returnFeeInfo',\n                        'returnLendingHistory',\n                        'returnMarginAccountSummary',\n                        'returnOpenLoanOffers',\n                        'returnOpenOrders',\n                        'returnOrderTrades',\n                        'returnTradableBalances',\n                        'returnTradeHistory',\n                        'sell',\n                        'toggleAutoRenew',\n                        'transferBalance',\n                        'withdraw',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.0015,\n                    'taker': 0.0025,\n                },\n                'funding': 0.0,\n            },\n            'limits': {\n                'amount': {\n                    'min': 0.00000001,\n                    'max': 1000000000,\n                },\n                'price': {\n                    'min': 0.00000001,\n                    'max': 1000000000,\n                },\n                'cost': {\n                    'min': 0.00000000,\n                    'max': 1000000000,\n                },\n            },\n            'precision': {\n                'amount': 8,\n                'price': 8,\n            },\n        });\n    }\n\n    calculateFee (symbol, type, side, amount, price, takerOrMaker = 'taker', params = {}) {\n        let market = this.markets[symbol];\n        let key = 'quote';\n        let rate = market[takerOrMaker];\n        let cost = parseFloat (this.costToPrecision (symbol, amount * rate));\n        if (side === 'sell') {\n            cost *= price;\n        } else {\n            key = 'base';\n        }\n        return {\n            'type': takerOrMaker,\n            'currency': market[key],\n            'rate': rate,\n            'cost': parseFloat (this.feeToPrecision (symbol, cost)),\n        };\n    }\n\n    commonCurrencyCode (currency) {\n        if (currency === 'BTM')\n            return 'Bitmark';\n        if (currency === 'STR')\n            return 'XLM';\n        return currency;\n    }\n\n    currencyId (currency) {\n        if (currency === 'Bitmark')\n            return 'BTM';\n        if (currency === 'XLM')\n            return 'STR';\n        return currency;\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '5m', since = undefined, limit = undefined) {\n        return [\n            ohlcv['date'] * 1000,\n            ohlcv['open'],\n            ohlcv['high'],\n            ohlcv['low'],\n            ohlcv['close'],\n            ohlcv['volume'],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '5m', since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        if (!since)\n            since = 0;\n        let request = {\n            'currencyPair': market['id'],\n            'period': this.timeframes[timeframe],\n            'start': parseInt (since / 1000),\n        };\n        if (limit)\n            request['end'] = this.sum (request['start'], limit * this.timeframes[timeframe]);\n        let response = await this.publicGetReturnChartData (this.extend (request, params));\n        return this.parseOHLCVs (response, market, timeframe, since, limit);\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetReturnTicker ();\n        let keys = Object.keys (markets);\n        let result = [];\n        for (let p = 0; p < keys.length; p++) {\n            let id = keys[p];\n            let market = markets[id];\n            let [ quote, base ] = id.split ('_');\n            base = this.commonCurrencyCode (base);\n            quote = this.commonCurrencyCode (quote);\n            let symbol = base + '/' + quote;\n            result.push (this.extend (this.fees['trading'], {\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'active': true,\n                'lot': this.limits['amount']['min'],\n                'info': market,\n            }));\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privatePostReturnCompleteBalances (this.extend ({\n            'account': 'all',\n        }, params));\n        let result = { 'info': balances };\n        let currencies = Object.keys (balances);\n        for (let c = 0; c < currencies.length; c++) {\n            let id = currencies[c];\n            let balance = balances[id];\n            let currency = this.commonCurrencyCode (id);\n            let account = {\n                'free': parseFloat (balance['available']),\n                'used': parseFloat (balance['onOrders']),\n                'total': 0.0,\n            };\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchFees (params = {}) {\n        await this.loadMarkets ();\n        let fees = await this.privatePostReturnFeeInfo ();\n        return {\n            'info': fees,\n            'maker': parseFloat (fees['makerFee']),\n            'taker': parseFloat (fees['takerFee']),\n            'withdraw': 0.0,\n        };\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetReturnOrderBook (this.extend ({\n            'currencyPair': this.marketId (symbol),\n            // 'depth': 100,\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high24hr']),\n            'low': parseFloat (ticker['low24hr']),\n            'bid': parseFloat (ticker['highestBid']),\n            'ask': parseFloat (ticker['lowestAsk']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': parseFloat (ticker['percentChange']),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['quoteVolume']),\n            'quoteVolume': parseFloat (ticker['baseVolume']),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetReturnTicker (params);\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchCurrencies (params = {}) {\n        let currencies = await this.publicGetReturnCurrencies (params);\n        let ids = Object.keys (currencies);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let currency = currencies[id];\n            // todo: will need to rethink the fees\n            // to add support for multiple withdrawal/deposit methods and\n            // differentiated fees for each particular method\n            let precision = 8; // default precision, todo: fix \"magic constants\"\n            let code = this.commonCurrencyCode (id);\n            let active = (currency['delisted'] === 0);\n            let status = (currency['disabled']) ? 'disabled' : 'ok';\n            if (status !== 'ok')\n                active = false;\n            result[code] = {\n                'id': id,\n                'code': code,\n                'info': currency,\n                'name': currency['name'],\n                'active': active,\n                'status': status,\n                'fee': currency['txFee'], // todo: redesign\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision),\n                        'max': Math.pow (10, precision),\n                    },\n                    'cost': {\n                        'min': undefined,\n                        'max': undefined,\n                    },\n                    'withdraw': {\n                        'min': currency['txFee'],\n                        'max': Math.pow (10, precision),\n                    },\n                },\n            };\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let tickers = await this.publicGetReturnTicker (params);\n        let ticker = tickers[market['id']];\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = this.parse8601 (trade['date']);\n        let symbol = undefined;\n        let base = undefined;\n        let quote = undefined;\n        if ((!market) && ('currencyPair' in trade)) {\n            let currencyPair = trade['currencyPair'];\n            if (currencyPair in this.markets_by_id) {\n                market = this.markets_by_id[currencyPair];\n            } else {\n                let parts = currencyPair.split ('_');\n                quote = parts[0];\n                base = parts[1];\n                symbol = base + '/' + quote;\n            }\n        }\n        if (market) {\n            symbol = market['symbol'];\n            base = market['base'];\n            quote = market['quote'];\n        }\n        let side = trade['type'];\n        let fee = undefined;\n        let cost = this.safeFloat (trade, 'total');\n        let amount = parseFloat (trade['amount']);\n        if ('fee' in trade) {\n            let rate = parseFloat (trade['fee']);\n            let feeCost = undefined;\n            let currency = undefined;\n            if (side === 'buy') {\n                currency = base;\n                feeCost = amount * rate;\n            } else {\n                currency = quote;\n                if (typeof cost !== 'undefined')\n                    feeCost = cost * rate;\n            }\n            fee = {\n                'type': undefined,\n                'rate': rate,\n                'cost': feeCost,\n                'currency': currency,\n            };\n        }\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': symbol,\n            'id': this.safeString (trade, 'tradeID'),\n            'order': this.safeString (trade, 'orderNumber'),\n            'type': 'limit',\n            'side': side,\n            'price': parseFloat (trade['rate']),\n            'amount': amount,\n            'cost': cost,\n            'fee': fee,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'currencyPair': market['id'],\n        };\n        if (since) {\n            request['start'] = parseInt (since / 1000);\n            request['end'] = this.seconds (); // last 50000 trades by default\n        }\n        let trades = await this.publicGetReturnTradeHistory (this.extend (request, params));\n        return this.parseTrades (trades, market, since, limit);\n    }\n\n    async fetchMyTrades (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        if (symbol)\n            market = this.market (symbol);\n        let pair = market ? market['id'] : 'all';\n        let request = { 'currencyPair': pair };\n        if (since) {\n            request['start'] = parseInt (since / 1000);\n            request['end'] = this.seconds ();\n        }\n        // limit is disabled (does not really work as expected)\n        // if (limit)\n        //     request['limit'] = parseInt (limit);\n        let response = await this.privatePostReturnTradeHistory (this.extend (request, params));\n        let result = [];\n        if (market) {\n            result = this.parseTrades (response, market);\n        } else {\n            if (response) {\n                let ids = Object.keys (response);\n                for (let i = 0; i < ids.length; i++) {\n                    let id = ids[i];\n                    let market = undefined;\n                    if (id in this.markets_by_id)\n                        market = this.markets_by_id[id];\n                    let trades = this.parseTrades (response[id], market);\n                    for (let j = 0; j < trades.length; j++) {\n                        result.push (trades[j]);\n                    }\n                }\n            }\n        }\n        return this.filterBySinceLimit (result, since, limit);\n    }\n\n    parseOrder (order, market = undefined) {\n        let timestamp = this.safeInteger (order, 'timestamp');\n        if (!timestamp)\n            timestamp = this.parse8601 (order['date']);\n        let trades = undefined;\n        if ('resultingTrades' in order)\n            trades = this.parseTrades (order['resultingTrades'], market);\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        let price = this.safeFloat (order, 'price');\n        let cost = this.safeFloat (order, 'total', 0.0);\n        let remaining = this.safeFloat (order, 'amount');\n        let amount = this.safeFloat (order, 'startingAmount', remaining);\n        let filled = undefined;\n        if (typeof amount !== 'undefined') {\n            if (typeof remaining !== 'undefined')\n                filled = amount - remaining;\n        }\n        if (typeof filled === 'undefined') {\n            if (typeof trades !== 'undefined') {\n                filled = 0;\n                cost = 0;\n                for (let i = 0; i < trades.length; i++) {\n                    let trade = trades[i];\n                    let tradeAmount = trade['amount'];\n                    let tradePrice = trade['price'];\n                    filled = this.sum (filled, tradeAmount);\n                    cost += tradePrice * tradeAmount;\n                }\n            }\n        }\n        return {\n            'info': order,\n            'id': order['orderNumber'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': order['status'],\n            'symbol': symbol,\n            'type': order['type'],\n            'side': order['side'],\n            'price': price,\n            'cost': cost,\n            'amount': amount,\n            'filled': filled,\n            'remaining': remaining,\n            'trades': trades,\n            'fee': undefined,\n        };\n    }\n\n    parseOpenOrders (orders, market, result = []) {\n        for (let i = 0; i < orders.length; i++) {\n            let order = orders[i];\n            let extended = this.extend (order, {\n                'status': 'open',\n                'type': 'limit',\n                'side': order['type'],\n                'price': order['rate'],\n            });\n            result.push (this.parseOrder (extended, market));\n        }\n        return result;\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        if (symbol)\n            market = this.market (symbol);\n        let pair = market ? market['id'] : 'all';\n        let response = await this.privatePostReturnOpenOrders (this.extend ({\n            'currencyPair': pair,\n        }));\n        let openOrders = [];\n        if (market) {\n            openOrders = this.parseOpenOrders (response, market, openOrders);\n        } else {\n            let marketIds = Object.keys (response);\n            for (let i = 0; i < marketIds.length; i++) {\n                let marketId = marketIds[i];\n                let orders = response[marketId];\n                let m = this.markets_by_id[marketId];\n                openOrders = this.parseOpenOrders (orders, m, openOrders);\n            }\n        }\n        for (let j = 0; j < openOrders.length; j++) {\n            this.orders[openOrders[j]['id']] = openOrders[j];\n        }\n        let openOrdersIndexedById = this.indexBy (openOrders, 'id');\n        let cachedOrderIds = Object.keys (this.orders);\n        let result = [];\n        for (let k = 0; k < cachedOrderIds.length; k++) {\n            let id = cachedOrderIds[k];\n            if (id in openOrdersIndexedById) {\n                this.orders[id] = this.extend (this.orders[id], openOrdersIndexedById[id]);\n            } else {\n                let order = this.orders[id];\n                if (order['status'] === 'open') {\n                    this.orders[id] = this.extend (order, {\n                        'status': 'closed',\n                        'cost': order['amount'] * order['price'],\n                        'filled': order['amount'],\n                        'remaining': 0.0,\n                    });\n                }\n            }\n            let order = this.orders[id];\n            if (market) {\n                if (order['symbol'] === symbol)\n                    result.push (order);\n            } else {\n                result.push (order);\n            }\n        }\n        return this.filterBySinceLimit (result, since, limit);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        let since = this.safeValue (params, 'since');\n        let limit = this.safeValue (params, 'limit');\n        let request = this.omit (params, [ 'since', 'limit' ]);\n        let orders = await this.fetchOrders (symbol, since, limit, request);\n        for (let i = 0; i < orders.length; i++) {\n            if (orders[i]['id'] === id)\n                return orders[i];\n        }\n        throw new OrderNotCached (this.id + ' order id ' + id.toString () + ' not found in cache');\n    }\n\n    filterOrdersByStatus (orders, status) {\n        let result = [];\n        for (let i = 0; i < orders.length; i++) {\n            if (orders[i]['status'] === status)\n                result.push (orders[i]);\n        }\n        return result;\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, since, limit, params);\n        return this.filterOrdersByStatus (orders, 'open');\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        let orders = await this.fetchOrders (symbol, since, limit, params);\n        return this.filterOrdersByStatus (orders, 'closed');\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        if (type === 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        await this.loadMarkets ();\n        let method = 'privatePost' + this.capitalize (side);\n        let market = this.market (symbol);\n        price = parseFloat (price);\n        amount = parseFloat (amount);\n        let response = await this[method] (this.extend ({\n            'currencyPair': market['id'],\n            'rate': this.priceToPrecision (symbol, price),\n            'amount': this.amountToPrecision (symbol, amount),\n        }, params));\n        let timestamp = this.milliseconds ();\n        let order = this.parseOrder (this.extend ({\n            'timestamp': timestamp,\n            'status': 'open',\n            'type': type,\n            'side': side,\n            'price': price,\n            'amount': amount,\n        }, response), market);\n        let id = order['id'];\n        this.orders[id] = order;\n        return this.extend ({ 'info': response }, order);\n    }\n\n    async editOrder (id, symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        price = parseFloat (price);\n        let request = {\n            'orderNumber': id,\n            'rate': this.priceToPrecision (symbol, price),\n        };\n        if (typeof amount !== 'undefined') {\n            amount = parseFloat (amount);\n            request['amount'] = this.amountToPrecision (symbol, amount);\n        }\n        let response = await this.privatePostMoveOrder (this.extend (request, params));\n        let result = undefined;\n        if (id in this.orders) {\n            this.orders[id]['status'] = 'canceled';\n            let newid = response['orderNumber'];\n            this.orders[newid] = this.extend (this.orders[id], {\n                'id': newid,\n                'price': price,\n                'status': 'open',\n            });\n            if (typeof amount !== 'undefined')\n                this.orders[newid]['amount'] = amount;\n            result = this.extend (this.orders[newid], { 'info': response });\n        } else {\n            let market = undefined;\n            if (symbol)\n                market = this.market (symbol);\n            result = this.parseOrder (response, market);\n            this.orders[result['id']] = result;\n        }\n        return result;\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = undefined;\n        try {\n            response = await this.privatePostCancelOrder (this.extend ({\n                'orderNumber': id,\n            }, params));\n            if (id in this.orders)\n                this.orders[id]['status'] = 'canceled';\n        } catch (e) {\n            if (this.last_http_response) {\n                if (this.last_http_response.indexOf ('Invalid order') >= 0)\n                    throw new OrderNotFound (this.id + ' cancelOrder() error: ' + this.last_http_response);\n            }\n            throw e;\n        }\n        return response;\n    }\n\n    async fetchOrderStatus (id, symbol = undefined) {\n        await this.loadMarkets ();\n        let orders = await this.fetchOpenOrders (symbol);\n        let indexed = this.indexBy (orders, 'id');\n        return (id in indexed) ? 'open' : 'closed';\n    }\n\n    async fetchOrderTrades (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let trades = await this.privatePostReturnOrderTrades (this.extend ({\n            'orderNumber': id,\n        }, params));\n        return this.parseTrades (trades);\n    }\n\n    async createDepositAddress (currency, params = {}) {\n        let currencyId = this.currencyId (currency);\n        let response = await this.privatePostGenerateNewAddress ({\n            'currency': currencyId,\n        });\n        let address = undefined;\n        if (response['success'] === 1)\n            address = this.safeString (response, 'response');\n        if (!address)\n            throw new ExchangeError (this.id + ' createDepositAddress failed: ' + this.last_http_response);\n        return {\n            'currency': currency,\n            'address': address,\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        let response = await this.privatePostReturnDepositAddresses ();\n        let currencyId = this.currencyId (currency);\n        let address = this.safeString (response, currencyId);\n        let status = address ? 'ok' : 'none';\n        return {\n            'currency': currency,\n            'address': address,\n            'status': status,\n            'info': response,\n        };\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let currencyId = this.currencyId (currency);\n        let request = {\n            'currency': currencyId,\n            'amount': amount,\n            'address': address,\n        };\n        if (tag)\n            request['paymentId'] = tag;\n        let result = await this.privatePostWithdraw (this.extend (request, params));\n        return {\n            'info': result,\n            'id': result['response'],\n        };\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        let query = this.extend ({ 'command': path }, params);\n        if (api === 'public') {\n            url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            query['nonce'] = this.nonce ();\n            body = this.urlencode (query);\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Key': this.apiKey,\n                'Sign': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code >= 400) {\n            if (body[0] === '{') {\n                let response = JSON.parse (body);\n                if ('error' in response) {\n                    let error = this.id + ' ' + body;\n                    if (response['error'].indexOf ('Total must be at least') >= 0) {\n                        throw new InvalidOrder (error);\n                    } else if (response['error'].indexOf ('Not enough') >= 0) {\n                        throw new InsufficientFunds (error);\n                    } else if (response['error'].indexOf ('Nonce must be greater') >= 0) {\n                        throw new ExchangeNotAvailable (error);\n                    }\n                }\n            }\n        }\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response) {\n            let error = this.id + ' ' + this.json (response);\n            let failed = response['error'].indexOf ('Not enough') >= 0;\n            if (failed)\n                throw new InsufficientFunds (error);\n            throw new ExchangeError (error);\n        }\n        return response;\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, OrderNotFound, InvalidOrder, InsufficientFunds, AuthenticationError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class qryptos extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'qryptos',\n            'name': 'QRYPTOS',\n            'countries': [ 'CN', 'TW' ],\n            'version': '2',\n            'rateLimit': 1000,\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n                'fetchOrder': true,\n                'fetchOrders': true,\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30953915-b1611dc0-a436-11e7-8947-c95bd5a42086.jpg',\n                'api': 'https://api.qryptos.com',\n                'www': 'https://www.qryptos.com',\n                'doc': [\n                    'https://developers.quoine.com',\n                    'https://developers.quoine.com/v2',\n                ],\n                'fees': 'https://qryptos.zendesk.com/hc/en-us/articles/115007858167-Fees',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'products',\n                        'products/{id}',\n                        'products/{id}/price_levels',\n                        'executions',\n                        'ir_ladders/{currency}',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'accounts/balance',\n                        'accounts/main_asset',\n                        'crypto_accounts',\n                        'executions/me',\n                        'fiat_accounts',\n                        'loan_bids',\n                        'loans',\n                        'orders',\n                        'orders/{id}',\n                        'orders/{id}/trades',\n                        'orders/{id}/executions',\n                        'trades',\n                        'trades/{id}/loans',\n                        'trading_accounts',\n                        'trading_accounts/{id}',\n                    ],\n                    'post': [\n                        'fiat_accounts',\n                        'loan_bids',\n                        'orders',\n                    ],\n                    'put': [\n                        'loan_bids/{id}/close',\n                        'loans/{id}',\n                        'orders/{id}',\n                        'orders/{id}/cancel',\n                        'trades/{id}',\n                        'trades/{id}/close',\n                        'trades/close_all',\n                        'trading_accounts/{id}',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetProducts ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['id'].toString ();\n            let base = market['base_currency'];\n            let quote = market['quoted_currency'];\n            let symbol = base + '/' + quote;\n            let maker = this.safeFloat (market, 'maker_fee');\n            let taker = this.safeFloat (market, 'taker_fee');\n            let active = !market['disabled'];\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'maker': maker,\n                'taker': taker,\n                'active': active,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privateGetAccountsBalance ();\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let total = parseFloat (balance['balance']);\n            let account = {\n                'free': total,\n                'used': 0.0,\n                'total': total,\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetProductsIdPriceLevels (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'buy_price_levels', 'sell_price_levels');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let last = undefined;\n        if ('last_traded_price' in ticker) {\n            if (ticker['last_traded_price']) {\n                let length = ticker['last_traded_price'].length;\n                if (length > 0)\n                    last = parseFloat (ticker['last_traded_price']);\n            }\n        }\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high_market_ask'),\n            'low': this.safeFloat (ticker, 'low_market_bid'),\n            'bid': this.safeFloat (ticker, 'market_bid'),\n            'ask': this.safeFloat (ticker, 'market_ask'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': last,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'volume_24h'),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetProducts (params);\n        let result = {};\n        for (let t = 0; t < tickers.length; t++) {\n            let ticker = tickers[t];\n            let base = ticker['base_currency'];\n            let quote = ticker['quoted_currency'];\n            let symbol = base + '/' + quote;\n            let market = this.markets[symbol];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetProductsId (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['created_at'] * 1000;\n        return {\n            'info': trade,\n            'id': trade['id'].toString (),\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['taker_side'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['quantity']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let request = {\n            'product_id': market['id'],\n        };\n        if (limit)\n            request['limit'] = limit;\n        let response = await this.publicGetExecutions (this.extend (request, params));\n        return this.parseTrades (response['models'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'order_type': type,\n            'product_id': this.marketId (symbol),\n            'side': side,\n            'quantity': amount,\n        };\n        if (type === 'limit')\n            order['price'] = price;\n        let response = await this.privatePostOrders (this.extend ({\n            'order': order,\n        }, params));\n        return this.parseOrder(response);\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let result = await this.privatePutOrdersIdCancel (this.extend ({\n            'id': id,\n        }, params));\n        let order = this.parseOrder (result);\n        if (order['status'] === 'closed')\n            throw new OrderNotFound (this.id + ' ' + this.json (order));\n        return order;\n    }\n\n    parseOrder (order) {\n        let timestamp = order['created_at'] * 1000;\n        let marketId = order['product_id'].toString ();\n        let market = this.marketsById[marketId];\n        let status = undefined;\n        if ('status' in order) {\n            if (order['status'] === 'live') {\n                status = 'open';\n            } else if (order['status'] === 'filled') {\n                status = 'closed';\n            } else if (order['status'] === 'cancelled') { // 'll' intended\n                status = 'canceled';\n            }\n        }\n        let amount = parseFloat (order['quantity']);\n        let filled = parseFloat (order['filled_quantity']);\n        let symbol = undefined;\n        if (market) {\n            symbol = market['symbol'];\n        }\n        return {\n            'id': order['id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'type': order['order_type'],\n            'status': status,\n            'symbol': symbol,\n            'side': order['side'],\n            'price': order['price'],\n            'amount': amount,\n            'filled': filled,\n            'remaining': amount - filled,\n            'trades': undefined,\n            'fee': {\n                'currency': undefined,\n                'cost': parseFloat (order['order_fee']),\n            },\n            'info': order,\n        };\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = await this.privateGetOrdersId (this.extend ({\n            'id': id,\n        }, params));\n        return this.parseOrder (order);\n    }\n\n    async fetchOrders (symbol = undefined, since = undefined, limit = undefined, params={}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        let request = {};\n        if (symbol) {\n            market = this.market (symbol);\n            request['product_id'] = market['id'];\n        }\n        let status = params['status'];\n        if (status === 'open') {\n            request['status'] = 'live';\n        } else if (status === 'closed') {\n            request['status'] = 'filled';\n        } else if (status === 'canceled') {\n            request['status'] = 'cancelled';\n        }\n        let result = await this.privateGetOrders (request);\n        let orders = result['models'];\n        return this.parseOrders (orders, market, since, limit);\n    }\n\n    fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        return this.fetchOrders (symbol, since, limit, this.extend ({ 'status': 'open' }, params));\n    }\n\n    fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        return this.fetchOrders (symbol, since, limit, this.extend ({ 'status': 'closed' }, params));\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        let response = undefined;\n        if (code === 200 || code === 404 || code === 422) {\n            if ((body[0] === '{') || (body[0] === '[')) {\n                response = JSON.parse (body);\n            } else {\n                // if not a JSON response\n                throw new ExchangeError (this.id + ' returned a non-JSON reply: ' + body);\n            }\n        }\n        if (code === 401) {\n            if (body === 'API Authentication failed') {\n                throw new AuthenticationError (body);\n            }\n        }\n        if (code === 404) {\n            if ('message' in response) {\n                if (response['message'] === 'Order not found') {\n                    throw new OrderNotFound (this.id + ' ' + body);\n                }\n            }\n        } else if (code === 422) {\n            if ('errors' in response) {\n                let errors = response['errors'];\n                if ('user' in errors) {\n                    let messages = errors['user'];\n                    if (messages.indexOf ('not_enough_free_balance') >= 0) {\n                        throw new InsufficientFunds (this.id + ' ' + body);\n                    }\n                } else if ('quantity' in errors) {\n                    let messages = errors['quantity'];\n                    if (messages.indexOf ('less_than_order_size') >= 0) {\n                        throw new InvalidOrder (this.id + ' ' + body);\n                    }\n                }\n            }\n        }\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        headers = {\n            'X-Quoine-API-Version': this.version,\n            'Content-Type': 'application/json',\n        };\n        if (api === 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            if (method === 'GET') {\n                if (Object.keys (query).length)\n                    url += '?' + this.urlencode (query);\n            } else if (Object.keys (query).length) {\n                body = this.json (query);\n            }\n            let nonce = this.nonce ();\n            let request = {\n                'path': url,\n                'nonce': nonce,\n                'token_id': this.apiKey,\n                'iat': Math.floor (nonce / 1000), // issued at\n            };\n            headers['X-Quoine-Auth'] = this.jwt (request, this.secret);\n        }\n        url = this.urls['api'] + url;\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n};\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class quadrigacx extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'quadrigacx',\n            'name': 'QuadrigaCX',\n            'countries': 'CA',\n            'rateLimit': 1000,\n            'version': 'v2',\n            'has': {\n                'CORS': true,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766825-98a6d0de-5ee7-11e7-9fa4-38e11a2c6f52.jpg',\n                'api': 'https://api.quadrigacx.com',\n                'www': 'https://www.quadrigacx.com',\n                'doc': 'https://www.quadrigacx.com/api_info',\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'uid': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'order_book',\n                        'ticker',\n                        'transactions',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'balance',\n                        'bitcoin_deposit_address',\n                        'bitcoin_withdrawal',\n                        'buy',\n                        'cancel_order',\n                        'ether_deposit_address',\n                        'ether_withdrawal',\n                        'lookup_order',\n                        'open_orders',\n                        'sell',\n                        'user_transactions',\n                    ],\n                },\n            },\n            'markets': {\n                'BTC/CAD': { 'id': 'btc_cad', 'symbol': 'BTC/CAD', 'base': 'BTC', 'quote': 'CAD', 'maker': 0.005, 'taker': 0.005 },\n                'BTC/USD': { 'id': 'btc_usd', 'symbol': 'BTC/USD', 'base': 'BTC', 'quote': 'USD', 'maker': 0.005, 'taker': 0.005 },\n                'ETH/BTC': { 'id': 'eth_btc', 'symbol': 'ETH/BTC', 'base': 'ETH', 'quote': 'BTC', 'maker': 0.002, 'taker': 0.002 },\n                'ETH/CAD': { 'id': 'eth_cad', 'symbol': 'ETH/CAD', 'base': 'ETH', 'quote': 'CAD', 'maker': 0.005, 'taker': 0.005 },\n                'LTC/CAD': { 'id': 'ltc_cad', 'symbol': 'LTC/CAD', 'base': 'LTC', 'quote': 'CAD', 'maker': 0.005, 'taker': 0.005 },\n                'BCH/CAD': { 'id': 'bch_cad', 'symbol': 'BCH/CAD', 'base': 'BCH', 'quote': 'CAD', 'maker': 0.005, 'taker': 0.005 },\n                'BTG/CAD': { 'id': 'btg_cad', 'symbol': 'BTG/CAD', 'base': 'BTG', 'quote': 'CAD', 'maker': 0.005, 'taker': 0.005 },\n            },\n        });\n    }\n\n    async fetchBalance (params = {}) {\n        let balances = await this.privatePostBalance ();\n        let result = { 'info': balances };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let lowercase = currency.toLowerCase ();\n            let account = {\n                'free': parseFloat (balances[lowercase + '_available']),\n                'used': parseFloat (balances[lowercase + '_reserved']),\n                'total': parseFloat (balances[lowercase + '_balance']),\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        let orderbook = await this.publicGetOrderBook (this.extend ({\n            'book': this.marketId (symbol),\n        }, params));\n        let timestamp = parseInt (orderbook['timestamp']) * 1000;\n        return this.parseOrderBook (orderbook, timestamp);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        let ticker = await this.publicGetTicker (this.extend ({\n            'book': this.marketId (symbol),\n        }, params));\n        let timestamp = parseInt (ticker['timestamp']) * 1000;\n        let vwap = parseFloat (ticker['vwap']);\n        let baseVolume = parseFloat (ticker['volume']);\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': vwap,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = parseInt (trade['date']) * 1000;\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': trade['tid'].toString (),\n            'order': undefined,\n            'type': undefined,\n            'side': trade['side'],\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        let market = this.market (symbol);\n        let response = await this.publicGetTransactions (this.extend ({\n            'book': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        let method = 'privatePost' + this.capitalize (side);\n        let order = {\n            'amount': amount,\n            'book': this.marketId (symbol),\n        };\n        if (type === 'limit')\n            order['price'] = price;\n        let response = await this[method] (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelOrder (this.extend ({\n            'id': id,\n        }, params));\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        let method = 'privatePost' + this.getCurrencyName (currency) + 'DepositAddress';\n        let response = await this[method] (params);\n        let address = undefined;\n        let status = undefined;\n        // [E|e]rror\n        if (response.indexOf ('rror') >= 0) {\n            status = 'error';\n        } else {\n            address = response;\n            status = 'ok';\n        }\n        return {\n            'currency': currency,\n            'address': address,\n            'status': status,\n            'info': this.last_http_response,\n        };\n    }\n\n    getCurrencyName (currency) {\n        if (currency === 'ETH')\n            return 'Ether';\n        if (currency === 'BTC')\n            return 'Bitcoin';\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let request = {\n            'amount': amount,\n            'address': address,\n        };\n        let method = 'privatePost' + this.getCurrencyName (currency) + 'Withdrawal';\n        let response = await this[method] (this.extend (request, params));\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + path;\n        if (api === 'public') {\n            url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let request = [ nonce.toString (), this.uid, this.apiKey ].join ('');\n            let signature = this.hmac (this.encode (request), this.encode (this.secret));\n            let query = this.extend ({\n                'key': this.apiKey,\n                'nonce': nonce,\n                'signature': signature,\n            }, params);\n            body = this.json (query);\n            headers = {\n                'Content-Type': 'application/json',\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (typeof response === 'string')\n            return response;\n        if ('error' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","'use strict';\n\n// ---------------------------------------------------------------------------\n\nconst qryptos = require ('./qryptos.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class quoinex extends qryptos {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'quoinex',\n            'name': 'QUOINEX',\n            'countries': [ 'JP', 'SG', 'VN' ],\n            'version': '2',\n            'rateLimit': 1000,\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/35047114-0e24ad4a-fbaa-11e7-96a9-69c1a756083b.jpg',\n                'api': 'https://api.quoine.com',\n                'www': 'https://quoinex.com/',\n                'doc': [\n                    'https://developers.quoine.com',\n                    'https://developers.quoine.com/v2',\n                ],\n                'fees': 'https://quoine.zendesk.com/hc/en-us/articles/115011281488-Fees',\n            },\n        });\n    }\n};\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nlet { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class southxchange extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'southxchange',\n            'name': 'SouthXchange',\n            'countries': 'AR', // Argentina\n            'rateLimit': 1000,\n            'has': {\n                'CORS': true,\n                'fetchTickers': true,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27838912-4f94ec8a-60f6-11e7-9e5d-bbf9bd50a559.jpg',\n                'api': 'https://www.southxchange.com/api',\n                'www': 'https://www.southxchange.com',\n                'doc': 'https://www.southxchange.com/Home/Api',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'markets',\n                        'price/{symbol}',\n                        'prices',\n                        'book/{symbol}',\n                        'trades/{symbol}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'cancelMarketOrders',\n                        'cancelOrder',\n                        'generatenewaddress',\n                        'listOrders',\n                        'listBalances',\n                        'placeOrder',\n                        'withdraw',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'maker': 0.2 / 100,\n                    'taker': 0.2 / 100,\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetMarkets ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let base = market[0];\n            let quote = market[1];\n            let symbol = base + '/' + quote;\n            let id = symbol;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privatePostListBalances ();\n        if (!balances)\n            throw new ExchangeError (this.id + ' fetchBalance got an unrecognized response');\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['Currency'];\n            let uppercase = currency.toUpperCase ();\n            let free = parseFloat (balance['Available']);\n            let used = parseFloat (balance['Unconfirmed']);\n            let total = this.sum (free, used);\n            let account = {\n                'free': free,\n                'used': used,\n                'total': total,\n            };\n            result[uppercase] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetBookSymbol (this.extend ({\n            'symbol': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook, undefined, 'BuyOrders', 'SellOrders', 'Price', 'Amount');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': undefined,\n            'low': undefined,\n            'bid': this.safeFloat (ticker, 'Bid'),\n            'ask': this.safeFloat (ticker, 'Ask'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'Last'),\n            'change': this.safeFloat (ticker, 'Variation24Hr'),\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': this.safeFloat (ticker, 'Volume24Hr'),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetPrices (params);\n        let tickers = this.indexBy (response, 'Market');\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let symbol = id;\n            let market = undefined;\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            }\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetPriceSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = trade['At'] * 1000;\n        return {\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'id': undefined,\n            'order': undefined,\n            'type': undefined,\n            'side': trade['Type'],\n            'price': trade['Price'],\n            'amount': trade['Amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTradesSymbol (this.extend ({\n            'symbol': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let order = {\n            'listingCurrency': market['base'],\n            'referenceCurrency': market['quote'],\n            'type': side,\n            'amount': amount,\n        };\n        if (type === 'limit')\n            order['limitPrice'] = price;\n        let response = await this.privatePostPlaceOrder (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response.toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelOrder (this.extend ({\n            'orderCode': id,\n        }, params));\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        let response = await this.privatePostWithdraw (this.extend ({\n            'currency': currency,\n            'address': address,\n            'amount': amount,\n        }, params));\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api === 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            query = this.extend ({\n                'key': this.apiKey,\n                'nonce': nonce,\n            }, query);\n            body = this.json (query);\n            headers = {\n                'Content-Type': 'application/json',\n                'Hash': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        return response;\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst foxbit = require ('./foxbit.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class surbitcoin extends foxbit {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'surbitcoin',\n            'name': 'SurBitcoin',\n            'countries': 'VE',\n            'has': {\n                'CORS': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27991511-f0a50194-6481-11e7-99b5-8f02932424cc.jpg',\n                'api': {\n                    'public': 'https://api.blinktrade.com/api',\n                    'private': 'https://api.blinktrade.com/tapi',\n                },\n                'www': 'https://surbitcoin.com',\n                'doc': 'https://blinktrade.com/docs',\n            },\n        });\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class therock extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'therock',\n            'name': 'TheRockTrading',\n            'countries': 'MT',\n            'rateLimit': 1000,\n            'version': 'v1',\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766869-75057fa2-5ee9-11e7-9a6f-13e641fa4707.jpg',\n                'api': 'https://api.therocktrading.com',\n                'www': 'https://therocktrading.com',\n                'doc': [\n                    'https://api.therocktrading.com/doc/v1/index.html',\n                    'https://api.therocktrading.com/doc/',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'funds/{id}/orderbook',\n                        'funds/{id}/ticker',\n                        'funds/{id}/trades',\n                        'funds/tickers',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'balances',\n                        'balances/{id}',\n                        'discounts',\n                        'discounts/{id}',\n                        'funds',\n                        'funds/{id}',\n                        'funds/{id}/trades',\n                        'funds/{fund_id}/orders',\n                        'funds/{fund_id}/orders/{id}',\n                        'funds/{fund_id}/position_balances',\n                        'funds/{fund_id}/positions',\n                        'funds/{fund_id}/positions/{id}',\n                        'transactions',\n                        'transactions/{id}',\n                        'withdraw_limits/{id}',\n                        'withdraw_limits',\n                    ],\n                    'post': [\n                        'atms/withdraw',\n                        'funds/{fund_id}/orders',\n                    ],\n                    'delete': [\n                        'funds/{fund_id}/orders/{id}',\n                        'funds/{fund_id}/orders/remove_all',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.02 / 100,\n                    'taker': 0.2 / 100,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.0005,\n                        'BCH': 0.0005,\n                        'PPC': 0.02,\n                        'ETH': 0.001,\n                        'ZEC': 0.001,\n                        'LTC': 0.002,\n                        'EUR': 2.5,  // worst-case scenario: https://therocktrading.com/en/pages/fees\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                        'BCH': 0,\n                        'PPC': 0,\n                        'ETH': 0,\n                        'ZEC': 0,\n                        'LTC': 0,\n                        'EUR': 0,\n                    },\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetFundsTickers ();\n        let result = [];\n        for (let p = 0; p < markets['tickers'].length; p++) {\n            let market = markets['tickers'][p];\n            let id = market['fund_id'];\n            let base = id.slice (0, 3);\n            let quote = id.slice (3);\n            let symbol = base + '/' + quote;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetBalances ();\n        let balances = response['balances'];\n        let result = { 'info': response };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let free = balance['trading_balance'];\n            let total = balance['balance'];\n            let used = total - free;\n            let account = {\n                'free': free,\n                'used': used,\n                'total': total,\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetFundsIdOrderbook (this.extend ({\n            'id': this.marketId (symbol),\n        }, params));\n        let timestamp = this.parse8601 (orderbook['date']);\n        return this.parseOrderBook (orderbook, timestamp, 'bids', 'asks', 'price', 'amount');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = this.parse8601 (ticker['date']);\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['bid']),\n            'ask': parseFloat (ticker['ask']),\n            'vwap': undefined,\n            'open': parseFloat (ticker['open']),\n            'close': parseFloat (ticker['close']),\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['volume_traded']),\n            'quoteVolume': parseFloat (ticker['volume']),\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetFundsTickers (params);\n        let tickers = this.indexBy (response['tickers'], 'fund_id');\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = this.markets_by_id[id];\n            let symbol = market['symbol'];\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let ticker = await this.publicGetFundsIdTicker (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTicker (ticker, market);\n    }\n\n    parseTrade (trade, market = undefined) {\n        if (!market)\n            market = this.markets_by_id[trade['fund_id']];\n        let timestamp = this.parse8601 (trade['date']);\n        return {\n            'info': trade,\n            'id': trade['id'].toString (),\n            'order': undefined,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': trade['side'],\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetFundsIdTrades (this.extend ({\n            'id': market['id'],\n        }, params));\n        return this.parseTrades (response['trades'], market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        if (type == 'market')\n            price = 0;\n        let response = await this.privatePostFundsFundIdOrders (this.extend ({\n            'fund_id': this.marketId (symbol),\n            'side': side,\n            'amount': amount,\n            'price': price,\n        }, params));\n        return {\n            'info': response,\n            'id': response['id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privateDeleteFundsFundIdOrdersId (this.extend ({\n            'id': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/' + this.version + '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ().toString ();\n            let auth = nonce + url;\n            headers = {\n                'X-TRT-KEY': this.apiKey,\n                'X-TRT-NONCE': nonce,\n                'X-TRT-SIGN': this.hmac (this.encode (auth), this.encode (this.secret), 'sha512'),\n            };\n            if (Object.keys (query).length) {\n                body = this.json (query);\n                headers['Content-Type'] = 'application/json';\n            }\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('errors' in response)\n            throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst liqui = require ('./liqui.js')\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class tidex extends liqui {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'tidex',\n            'name': 'Tidex',\n            'countries': 'UK',\n            'rateLimit': 2000,\n            'version': '3',\n            'has': {\n                // 'CORS': false,\n                // 'fetchTickers': true\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30781780-03149dc4-a12e-11e7-82bb-313b269d24d4.jpg',\n                'api': {\n                    'public': 'https://api.tidex.com/api',\n                    'private': 'https://api.tidex.com/tapi',\n                },\n                'www': 'https://tidex.com',\n                'doc': 'https://tidex.com/public-api',\n                'fees': [\n                    'https://tidex.com/assets-spec',\n                    'https://tidex.com/pairs-spec',\n                ],\n            },\n            'fees': {\n                'trading': {\n                    'tierBased': false,\n                    'percentage': true,\n                    'taker': 0.1 / 100,\n                    'maker': 0.1 / 100,\n                },\n                'funding': {\n                    'tierBased': false,\n                    'percentage': false,\n                    'withdraw': {\n                        'BTC': 0.002,\n                        'LTC': 0.001,\n                        'ETH': 0.01,\n                        'DASH': 0.01,\n                        'DOGE': 0.01,\n                        'BTS': 5.0,\n                        'STEEM': 0.01,\n                        'WAVES': 0.01,\n                        'WCT': 0.01,\n                        'WBTC': 0.0001,\n                        'INCNT': 0.1,\n                        'B': 0.1,\n                        'MRT': 1.0,\n                        'MER': 5.0,\n                        'AQUA': 0.001,\n                        'RBX': 100.0,\n                        'TKS': 0.1,\n                        'WUSD': 0.1,\n                        'WEUR': 0.1,\n                        'WGO': 1.0,\n                        'GNT': 1.0,\n                        'EDG': 1.0,\n                        'RLC': 0.3,\n                        'ICN': 0.3,\n                        'WINGS': 1.0,\n                        'VSL': 1.0,\n                        'TIME': 0.01,\n                        'TAAS': 0.3,\n                        'KOLION': 0.1,\n                        'RIDDLE': 10.0,\n                        'ANT': 0.1,\n                        'EFYT': 0.1,\n                        'MGO': 0.5,\n                        'wETT': 1.0,\n                        'eETT': 1.0,\n                        'QRL': 1.0,\n                        'eMGO': 1.0,\n                        'BNT': 1.0,\n                        'SNM': 10.0,\n                        'ZRC': 0.1,\n                        'SNT': 30.0,\n                        'MCO': 0.3,\n                        'STORJ': 1.0,\n                        'EOS': 0.3,\n                        'WGR': 3.0,\n                        'STA': 0.1,\n                        'PBT': 0.0001,\n                        'BCH': 0.00125,\n                        'wSUR': 0.05,\n                        'SUR': 0.05,\n                        'MSP': 5.0,\n                        'InPay': 0.5,\n                        'MTL': 0.1,\n                        'AHT': 0.2,\n                        'PING': 1.0,\n                        'EOT': 0.2,\n                        'AE': 3.0,\n                        'PIX': 10.0,\n                        'CREDO': 30.0,\n                        'LIFE': 1000.0,\n                        'MTH': 5.0,\n                        'BMC': 1.0,\n                        'TRCT': 5.0,\n                        'KNC': 1.0,\n                        'MSD': 0.2,\n                        'SUB': 10.0,\n                        'ENJ': 20.0,\n                        'EVX': 1.0,\n                        'OCL': 3.0,\n                        'ENG': 2.0,\n                        'TDX': 1.0,\n                        'LA': 2.0,\n                        'PRG': 0.5,\n                        'ICOS': 0.03,\n                        'USDT': 40.0,\n                        'ARN': 2.0,\n                        'RYZ': 10.0,\n                        'B2B': 1.0,\n                        'CAT': 10.0,\n                        'SNOV': 25.0,\n                        'DRGN': 3.0,\n                        'TIE': 20.0,\n                        'TRX': 50.0,\n                        'WAX': 5.0,\n                        'AGI': 5.0,\n                        'VEE': 20.0,\n                    },\n                    'deposit': {\n                        'BTC': 0,\n                        'ETH': 0,\n                        'LTC': 0,\n                        'DOGE': 0,\n                        'ICN': 0,\n                        'DASH': 0,\n                        'GNO': 0,\n                        'EOS': 0,\n                        'BCH': 0,\n                        'USDT': 0,\n                    },\n                },\n            },\n        });\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst foxbit = require ('./foxbit.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class urdubit extends foxbit {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'urdubit',\n            'name': 'UrduBit',\n            'countries': 'PK',\n            'has': {\n                'CORS': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27991453-156bf3ae-6480-11e7-82eb-7295fe1b5bb4.jpg',\n                'api': {\n                    'public': 'https://api.blinktrade.com/api',\n                    'private': 'https://api.blinktrade.com/tapi',\n                },\n                'www': 'https://urdubit.com',\n                'doc': 'https://blinktrade.com/docs',\n            },\n        });\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class vaultoro extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'vaultoro',\n            'name': 'Vaultoro',\n            'countries': 'CH',\n            'rateLimit': 1000,\n            'version': '1',\n            'has': {\n                'CORS': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766880-f205e870-5ee9-11e7-8fe2-0d5b15880752.jpg',\n                'api': 'https://api.vaultoro.com',\n                'www': 'https://www.vaultoro.com',\n                'doc': 'https://api.vaultoro.com',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'bidandask',\n                        'buyorders',\n                        'latest',\n                        'latesttrades',\n                        'markets',\n                        'orderbook',\n                        'sellorders',\n                        'transactions/day',\n                        'transactions/hour',\n                        'transactions/month',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'balance',\n                        'mytrades',\n                        'orders',\n                    ],\n                    'post': [\n                        'buy/{symbol}/{type}',\n                        'cancel/{id}',\n                        'sell/{symbol}/{type}',\n                        'withdraw',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let result = [];\n        let markets = await this.publicGetMarkets ();\n        let market = markets['data'];\n        let base = market['BaseCurrency'];\n        let quote = market['MarketCurrency'];\n        let symbol = base + '/' + quote;\n        let baseId = base;\n        let quoteId = quote;\n        let id = market['MarketName'];\n        result.push ({\n            'id': id,\n            'symbol': symbol,\n            'base': base,\n            'quote': quote,\n            'baseId': baseId,\n            'quoteId': quoteId,\n            'info': market,\n        });\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privateGetBalance ();\n        let balances = response['data'];\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency_code'];\n            let uppercase = currency.toUpperCase ();\n            let free = balance['cash'];\n            let used = balance['reserved'];\n            let total = this.sum (free, used);\n            let account = {\n                'free': free,\n                'used': used,\n                'total': total,\n            };\n            result[uppercase] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetOrderbook (params);\n        let orderbook = {\n            'bids': response['data'][0]['b'],\n            'asks': response['data'][1]['s'],\n        };\n        let result = this.parseOrderBook (orderbook, undefined, 'bids', 'asks', 'Gold_Price', 'Gold_Amount');\n        result['bids'] = this.sortBy (result['bids'], 0, true);\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let quote = await this.publicGetBidandask (params);\n        let bidsLength = quote['bids'].length;\n        let bid = quote['bids'][bidsLength - 1];\n        let ask = quote['asks'][0];\n        let response = await this.publicGetMarkets (params);\n        let ticker = response['data'];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['24hHigh']),\n            'low': parseFloat (ticker['24hLow']),\n            'bid': bid[0],\n            'ask': ask[0],\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['LastPrice']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': undefined,\n            'quoteVolume': parseFloat (ticker['24hVolume']),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market) {\n        let timestamp = this.parse8601 (trade['Time']);\n        return {\n            'id': undefined,\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'order': undefined,\n            'type': undefined,\n            'side': undefined,\n            'price': trade['Gold_Price'],\n            'amount': trade['Gold_Amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTransactionsDay (params);\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let method = 'privatePost' + this.capitalize (side) + 'SymbolType';\n        let response = await this[method] (this.extend ({\n            'symbol': market['quoteId'].toLowerCase (),\n            'type': type,\n            'gld': amount,\n            'price': price || 1,\n        }, params));\n        return {\n            'info': response,\n            'id': response['data']['Order_ID'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        return await this.privatePostCancelId (this.extend ({\n            'id': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/';\n        if (api == 'public') {\n            url += path;\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            url += this.version + '/' + this.implodeParams (path, params);\n            let query = this.extend ({\n                'nonce': nonce,\n                'apikey': this.apiKey,\n            }, this.omit (params, this.extractParams (path)));\n            url += '?' + this.urlencode (query);\n            headers = {\n                'Content-Type': 'application/json',\n                'X-Signature': this.hmac (this.encode (url), this.encode (this.secret)),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst foxbit = require ('./foxbit.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class vbtc extends foxbit {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'vbtc',\n            'name': 'VBTC',\n            'countries': 'VN',\n            'has': {\n                'CORS': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27991481-1f53d1d8-6481-11e7-884e-21d17e7939db.jpg',\n                'api': {\n                    'public': 'https://api.blinktrade.com/api',\n                    'private': 'https://api.blinktrade.com/tapi',\n                },\n                'www': 'https://vbtc.exchange',\n                'doc': 'https://blinktrade.com/docs',\n            },\n        });\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class virwox extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'virwox',\n            'name': 'VirWoX',\n            'countries': [ 'AT', 'EU' ],\n            'rateLimit': 1000,\n            'has': {\n                'CORS': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766894-6da9d360-5eea-11e7-90aa-41f2711b7405.jpg',\n                'api': {\n                    'public': 'http://api.virwox.com/api/json.php',\n                    'private': 'https://www.virwox.com/api/trading.php',\n                },\n                'www': 'https://www.virwox.com',\n                'doc': 'https://www.virwox.com/developers.php',\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': false,\n                'login': true,\n                'password': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'getInstruments',\n                        'getBestPrices',\n                        'getMarketDepth',\n                        'estimateMarketOrder',\n                        'getTradedPriceVolume',\n                        'getRawTradeData',\n                        'getStatistics',\n                        'getTerminalList',\n                        'getGridList',\n                        'getGridStatistics',\n                    ],\n                    'post': [\n                        'getInstruments',\n                        'getBestPrices',\n                        'getMarketDepth',\n                        'estimateMarketOrder',\n                        'getTradedPriceVolume',\n                        'getRawTradeData',\n                        'getStatistics',\n                        'getTerminalList',\n                        'getGridList',\n                        'getGridStatistics',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'cancelOrder',\n                        'getBalances',\n                        'getCommissionDiscount',\n                        'getOrders',\n                        'getTransactions',\n                        'placeOrder',\n                    ],\n                    'post': [\n                        'cancelOrder',\n                        'getBalances',\n                        'getCommissionDiscount',\n                        'getOrders',\n                        'getTransactions',\n                        'placeOrder',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetInstruments ();\n        let keys = Object.keys (markets['result']);\n        let result = [];\n        for (let p = 0; p < keys.length; p++) {\n            let market = markets['result'][keys[p]];\n            let id = market['instrumentID'];\n            let symbol = market['symbol'];\n            let base = market['longCurrency'];\n            let quote = market['shortCurrency'];\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetBalances ();\n        let balances = response['result']['accountList'];\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['currency'];\n            let total = balance['balance'];\n            let account = {\n                'free': total,\n                'used': 0.0,\n                'total': total,\n            };\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchMarketPrice (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicPostGetBestPrices (this.extend ({\n            'symbols': [ symbol ],\n        }, params));\n        let result = response['result'];\n        return {\n            'bid': this.safeFloat (result[0], 'bestBuyPrice'),\n            'ask': this.safeFloat (result[0], 'bestSellPrice'),\n        };\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicPostGetMarketDepth (this.extend ({\n            'symbols': [ symbol ],\n            'buyDepth': 100,\n            'sellDepth': 100,\n        }, params));\n        let orderbook = response['result'][0];\n        return this.parseOrderBook (orderbook, undefined, 'buy', 'sell', 'price', 'volume');\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let end = this.milliseconds ();\n        let start = end - 86400000;\n        let response = await this.publicGetTradedPriceVolume (this.extend ({\n            'instrument': symbol,\n            'endDate': this.YmdHMS (end),\n            'startDate': this.YmdHMS (start),\n            'HLOC': 1,\n        }, params));\n        let marketPrice = await this.fetchMarketPrice (symbol, params);\n        let tickers = response['result']['priceVolumeList'];\n        let keys = Object.keys (tickers);\n        let length = keys.length;\n        let lastKey = keys[length - 1];\n        let ticker = tickers[lastKey];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': marketPrice['bid'],\n            'ask': marketPrice['ask'],\n            'vwap': undefined,\n            'open': parseFloat (ticker['open']),\n            'close': parseFloat (ticker['close']),\n            'first': undefined,\n            'last': undefined,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['longVolume']),\n            'quoteVolume': parseFloat (ticker['shortVolume']),\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, symbol = undefined) {\n        let sec = this.safeInteger (trade, 'time');\n        let timestamp = sec * 1000;\n        return {\n            'id': trade['tid'],\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'order': undefined,\n            'symbol': symbol,\n            'type': undefined,\n            'side': undefined,\n            'price': this.safeFloat (trade, 'price'),\n            'amount': this.safeFloat (trade, 'vol'),\n            'fee': undefined,\n            'info': trade,\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.publicGetRawTradeData (this.extend ({\n            'instrument': symbol,\n            'timespan': 3600,\n        }, params));\n        let result = response['result'];\n        let trades = result['data'];\n        return this.parseTrades (trades, symbol);\n    }\n\n    async createOrder (market, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let order = {\n            'instrument': this.symbol (market),\n            'orderType': side.toUpperCase (),\n            'amount': amount,\n        };\n        if (type == 'limit')\n            order['price'] = price;\n        let response = await this.privatePostPlaceOrder (this.extend (order, params));\n        return {\n            'info': response,\n            'id': response['orderID'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelOrder (this.extend ({\n            'orderID': id,\n        }, params));\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        let auth = {};\n        if (api == 'private') {\n            this.checkRequiredCredentials ();\n            auth['key'] = this.apiKey;\n            auth['user'] = this.login;\n            auth['pass'] = this.password;\n        }\n        let nonce = this.nonce ();\n        if (method == 'GET') {\n            url += '?' + this.urlencode (this.extend ({\n                'method': path,\n                'id': nonce,\n            }, auth, params));\n        } else {\n            headers = { 'Content-Type': 'application/json' };\n            body = this.json ({\n                'method': path,\n                'params': this.extend (auth, params),\n                'id': nonce,\n            });\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code == 200) {\n            if ((body[0] == '{') || (body[0] == '[')) {\n                let response = JSON.parse (body);\n                if ('result' in response) {\n                    let result = response['result'];\n                    if ('errorCode' in result) {\n                        let errorCode = result['errorCode'];\n                        if (errorCode != 'OK') {\n                            throw new ExchangeError (this.id + ' error returned: ' + body);\n                        }\n                    }\n                } else {\n                    throw new ExchangeError (this.id + ' malformed response: no result in response: ' + body);\n                }\n            } else {\n                // if not a JSON response\n                throw new ExchangeError (this.id + ' returned a non-JSON reply: ' + body);\n            }\n        }\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst liqui = require ('./liqui.js');\nconst { ExchangeError, InsufficientFunds, OrderNotFound, DDoSProtection } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class wex extends liqui {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'wex',\n            'name': 'WEX',\n            'countries': 'NZ', // New Zealand\n            'version': '3',\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/30652751-d74ec8f8-9e31-11e7-98c5-71469fcef03e.jpg',\n                'api': {\n                    'public': 'https://wex.nz/api',\n                    'private': 'https://wex.nz/tapi',\n                },\n                'www': 'https://wex.nz',\n                'doc': [\n                    'https://wex.nz/api/3/docs',\n                    'https://wex.nz/tapi/docs',\n                ],\n                'fees': 'https://wex.nz/fees',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'info',\n                        'ticker/{pair}',\n                        'depth/{pair}',\n                        'trades/{pair}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'getInfo',\n                        'Trade',\n                        'ActiveOrders',\n                        'OrderInfo',\n                        'CancelOrder',\n                        'TradeHistory',\n                        'TransHistory',\n                        'CoinDepositAddress',\n                        'WithdrawCoin',\n                        'CreateCoupon',\n                        'RedeemCoupon',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.2 / 100,\n                    'taker': 0.2 / 100,\n                },\n                'funding': {\n                    'withdraw': {\n                        'BTC': 0.001,\n                        'LTC': 0.001,\n                        'NMC': 0.1,\n                        'NVC': 0.1,\n                        'PPC': 0.1,\n                        'DASH': 0.001,\n                        'ETH': 0.003,\n                        'BCH': 0.001,\n                        'ZEC': 0.001,\n                    },\n                },\n            },\n        });\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = ticker['updated'] * 1000;\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': this.safeFloat (ticker, 'high'),\n            'low': this.safeFloat (ticker, 'low'),\n            'bid': this.safeFloat (ticker, 'sell'),\n            'ask': this.safeFloat (ticker, 'buy'),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': this.safeFloat (ticker, 'last'),\n            'change': undefined,\n            'percentage': undefined,\n            'average': this.safeFloat (ticker, 'avg'),\n            'baseVolume': this.safeFloat (ticker, 'vol_cur'),\n            'quoteVolume': this.safeFloat (ticker, 'vol'),\n            'info': ticker,\n        };\n    }\n\n    handleErrors (code, reason, url, method, headers, body) {\n        if (code == 200) {\n            if (body[0] != '{') {\n                // response is not JSON\n                throw new ExchangeError (this.id + ' returned a non-JSON reply: ' + body);\n            }\n            let response = JSON.parse (body);\n            if ('success' in response) {\n                if (!response['success']) {\n                    let error = this.safeValue (response, 'error');\n                    if (!error) {\n                        throw new ExchangeError (this.id + ' returned a malformed error: ' + body);\n                    } else if (error == 'bad status') {\n                        throw new OrderNotFound (this.id + ' ' + error);\n                    } else if (error.indexOf ('It is not enough') >= 0) {\n                        throw new InsufficientFunds (this.id + ' ' + error);\n                    } else if (error == 'Requests too often') {\n                        throw new DDoSProtection (this.id + ' ' + error);\n                    } else if (error == 'not available') {\n                        throw new DDoSProtection (this.id + ' ' + error);\n                    } else if (error == 'external service unavailable') {\n                        throw new DDoSProtection (this.id + ' ' + error);\n                    // that's what fetchOpenOrders return if no open orders (fix for #489)\n                    } else if (error != 'no orders') {\n                        throw new ExchangeError (this.id + ' ' + error);\n                    }\n                }\n            }\n        }\n    }\n\n    request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        return this.fetch2 (path, api, method, params, headers, body);\n    }\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError, NotSupported, AuthenticationError } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class xbtce extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'xbtce',\n            'name': 'xBTCe',\n            'countries': 'RU',\n            'rateLimit': 2000, // responses are cached every 2 seconds\n            'version': 'v1',\n            'has': {\n                'publicAPI': false,\n                'CORS': false,\n                'fetchTickers': true,\n                'fetchOHLCV': false,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28059414-e235970c-662c-11e7-8c3a-08e31f78684b.jpg',\n                'api': 'https://cryptottlivewebapi.xbtce.net:8443/api',\n                'www': 'https://www.xbtce.com',\n                'doc': [\n                    'https://www.xbtce.com/tradeapi',\n                    'https://support.xbtce.info/Knowledgebase/Article/View/52/25/xbtce-exchange-api',\n                ],\n            },\n            'requiredCredentials': {\n                'apiKey': true,\n                'secret': true,\n                'uid': true,\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'currency',\n                        'currency/{filter}',\n                        'level2',\n                        'level2/{filter}',\n                        'quotehistory/{symbol}/{periodicity}/bars/ask',\n                        'quotehistory/{symbol}/{periodicity}/bars/bid',\n                        'quotehistory/{symbol}/level2',\n                        'quotehistory/{symbol}/ticks',\n                        'symbol',\n                        'symbol/{filter}',\n                        'tick',\n                        'tick/{filter}',\n                        'ticker',\n                        'ticker/{filter}',\n                        'tradesession',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'tradeserverinfo',\n                        'tradesession',\n                        'currency',\n                        'currency/{filter}',\n                        'level2',\n                        'level2/{filter}',\n                        'symbol',\n                        'symbol/{filter}',\n                        'tick',\n                        'tick/{filter}',\n                        'account',\n                        'asset',\n                        'asset/{id}',\n                        'position',\n                        'position/{id}',\n                        'trade',\n                        'trade/{id}',\n                        'quotehistory/{symbol}/{periodicity}/bars/ask',\n                        'quotehistory/{symbol}/{periodicity}/bars/ask/info',\n                        'quotehistory/{symbol}/{periodicity}/bars/bid',\n                        'quotehistory/{symbol}/{periodicity}/bars/bid/info',\n                        'quotehistory/{symbol}/level2',\n                        'quotehistory/{symbol}/level2/info',\n                        'quotehistory/{symbol}/periodicities',\n                        'quotehistory/{symbol}/ticks',\n                        'quotehistory/{symbol}/ticks/info',\n                        'quotehistory/cache/{symbol}/{periodicity}/bars/ask',\n                        'quotehistory/cache/{symbol}/{periodicity}/bars/bid',\n                        'quotehistory/cache/{symbol}/level2',\n                        'quotehistory/cache/{symbol}/ticks',\n                        'quotehistory/symbols',\n                        'quotehistory/version',\n                    ],\n                    'post': [\n                        'trade',\n                        'tradehistory',\n                    ],\n                    'put': [\n                        'trade',\n                    ],\n                    'delete': [\n                        'trade',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.privateGetSymbol ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['Symbol'];\n            let base = market['MarginCurrency'];\n            let quote = market['ProfitCurrency'];\n            if (base == 'DSH')\n                base = 'DASH';\n            let symbol = base + '/' + quote;\n            symbol = market['IsTradeAllowed'] ? symbol : id;\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let balances = await this.privateGetAsset ();\n        let result = { 'info': balances };\n        for (let b = 0; b < balances.length; b++) {\n            let balance = balances[b];\n            let currency = balance['Currency'];\n            let uppercase = currency.toUpperCase ();\n            // xbtce names DASH incorrectly as DSH\n            if (uppercase == 'DSH')\n                uppercase = 'DASH';\n            let account = {\n                'free': balance['FreeAmount'],\n                'used': balance['LockedAmount'],\n                'total': balance['Amount'],\n            };\n            result[uppercase] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let orderbook = await this.privateGetLevel2Filter (this.extend ({\n            'filter': market['id'],\n        }, params));\n        orderbook = orderbook[0];\n        let timestamp = orderbook['Timestamp'];\n        return this.parseOrderBook (orderbook, timestamp, 'Bids', 'Asks', 'Price', 'Volume');\n    }\n\n    parseTicker (ticker, market = undefined) {\n        let timestamp = 0;\n        let last = undefined;\n        if ('LastBuyTimestamp' in ticker)\n            if (timestamp < ticker['LastBuyTimestamp']) {\n                timestamp = ticker['LastBuyTimestamp'];\n                last = ticker['LastBuyPrice'];\n            }\n        if ('LastSellTimestamp' in ticker)\n            if (timestamp < ticker['LastSellTimestamp']) {\n                timestamp = ticker['LastSellTimestamp'];\n                last = ticker['LastSellPrice'];\n            }\n        if (!timestamp)\n            timestamp = this.milliseconds ();\n        let symbol = undefined;\n        if (market)\n            symbol = market['symbol'];\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': ticker['DailyBestBuyPrice'],\n            'low': ticker['DailyBestSellPrice'],\n            'bid': ticker['BestBid'],\n            'ask': ticker['BestAsk'],\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': last,\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': ticker['DailyTradedTotalVolume'],\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    async fetchTickers (symbols = undefined, params = {}) {\n        await this.loadMarkets ();\n        let tickers = await this.publicGetTicker (params);\n        tickers = this.indexBy (tickers, 'Symbol');\n        let ids = Object.keys (tickers);\n        let result = {};\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let market = undefined;\n            let symbol = undefined;\n            if (id in this.markets_by_id) {\n                market = this.markets_by_id[id];\n                symbol = market['symbol'];\n            } else {\n                let base = id.slice (0, 3);\n                let quote = id.slice (3, 6);\n                if (base == 'DSH')\n                    base = 'DASH';\n                if (quote == 'DSH')\n                    quote = 'DASH';\n                symbol = base + '/' + quote;\n            }\n            let ticker = tickers[id];\n            result[symbol] = this.parseTicker (ticker, market);\n        }\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let tickers = await this.publicGetTickerFilter (this.extend ({\n            'filter': market['id'],\n        }, params));\n        let length = tickers.length;\n        if (length < 1)\n            throw new ExchangeError (this.id + ' fetchTicker returned empty response, xBTCe public API error');\n        tickers = this.indexBy (tickers, 'Symbol');\n        let ticker = tickers[market['id']];\n        return this.parseTicker (ticker, market);\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        // no method for trades?\n        return await this.privateGetTrade (params);\n    }\n\n    parseOHLCV (ohlcv, market = undefined, timeframe = '1m', since = undefined, limit = undefined) {\n        return [\n            ohlcv['Timestamp'],\n            ohlcv['Open'],\n            ohlcv['High'],\n            ohlcv['Low'],\n            ohlcv['Close'],\n            ohlcv['Volume'],\n        ];\n    }\n\n    async fetchOHLCV (symbol, timeframe = '1m', since = undefined, limit = undefined, params = {}) {\n        throw new NotSupported (this.id + ' fetchOHLCV is disabled by the exchange');\n        let minutes = parseInt (timeframe / 60); // 1 minute by default\n        let periodicity = minutes.toString ();\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        if (!since)\n            since = this.seconds () - 86400 * 7; // last day by defulat\n        if (!limit)\n            limit = 1000; // default\n        let response = await this.privateGetQuotehistorySymbolPeriodicityBarsBid (this.extend ({\n            'symbol': market['id'],\n            'periodicity': periodicity,\n            'timestamp': since,\n            'count': limit,\n        }, params));\n        return this.parseOHLCVs (response['Bars'], market, timeframe, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        if (type == 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let response = await this.tapiPostTrade (this.extend ({\n            'pair': this.marketId (symbol),\n            'type': side,\n            'amount': amount,\n            'rate': price,\n        }, params));\n        return {\n            'info': response,\n            'id': response['Id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privateDeleteTrade (this.extend ({\n            'Type': 'Cancel',\n            'Id': id,\n        }, params));\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        if (!this.apiKey)\n            throw new AuthenticationError (this.id + ' requires apiKey for all requests, their public API is always busy');\n        if (!this.uid)\n            throw new AuthenticationError (this.id + ' requires uid property for authentication and trading, their public API is always busy');\n        let url = this.urls['api'] + '/' + this.version;\n        if (api == 'public')\n            url += '/' + api;\n        url += '/' + this.implodeParams (path, params);\n        let query = this.omit (params, this.extractParams (path));\n        if (api == 'public') {\n            if (Object.keys (query).length)\n                url += '?' + this.urlencode (query);\n        } else {\n            this.checkRequiredCredentials ();\n            headers = { 'Accept-Encoding': 'gzip, deflate' };\n            let nonce = this.nonce ().toString ();\n            if (method == 'POST') {\n                if (Object.keys (query).length) {\n                    headers['Content-Type'] = 'application/json';\n                    body = this.json (query);\n                } else {\n                    url += '?' + this.urlencode (query);\n                }\n            }\n            let auth = nonce + this.uid + this.apiKey + method + url;\n            if (body)\n                auth += body;\n            let signature = this.hmac (this.encode (auth), this.encode (this.secret), 'sha256', 'base64');\n            let credentials = this.uid + ':' + this.apiKey + ':' + nonce + ':' + this.binaryToString (signature);\n            headers['Authorization'] = 'HMAC ' + credentials;\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n}\n","'use strict';\n\n// ---------------------------------------------------------------------------\n\nconst liqui = require ('./liqui.js');\nconst { ExchangeError, InsufficientFunds, DDoSProtection } = require ('./base/errors');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class yobit extends liqui {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'yobit',\n            'name': 'YoBit',\n            'countries': 'RU',\n            'rateLimit': 3000, // responses are cached every 2 seconds\n            'version': '3',\n            'has': {\n                'CORS': false,\n                'withdraw': true\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766910-cdcbfdae-5eea-11e7-9859-03fea873272d.jpg',\n                'api': {\n                    'public': 'https://yobit.net/api',\n                    'private': 'https://yobit.net/tapi',\n                },\n                'www': 'https://www.yobit.net',\n                'doc': 'https://www.yobit.net/en/api/',\n                'fees': 'https://www.yobit.net/en/fees/',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'depth/{pair}',\n                        'info',\n                        'ticker/{pair}',\n                        'trades/{pair}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'ActiveOrders',\n                        'CancelOrder',\n                        'GetDepositAddress',\n                        'getInfo',\n                        'OrderInfo',\n                        'Trade',\n                        'TradeHistory',\n                        'WithdrawCoinsToAddress',\n                    ],\n                },\n            },\n            'fees': {\n                'trading': {\n                    'maker': 0.002,\n                    'taker': 0.002,\n                },\n                'funding': 0.0,\n                'withdraw': 0.0005,\n            },\n        });\n    }\n\n    commonCurrencyCode (currency) {\n        let substitutions = {\n            'AIR': 'AirCoin',\n            'ANI': 'ANICoin',\n            'ANT': 'AntsCoin',\n            'ATM': 'Autumncoin',\n            'BCC': 'BCH',\n            'BTS': 'Bitshares2',\n            'DCT': 'Discount',\n            'DGD': 'DarkGoldCoin',\n            'ICN': 'iCoin',\n            'LIZI': 'LiZi',\n            'LUN': 'LunarCoin',\n            'NAV': 'NavajoCoin',\n            'OMG': 'OMGame',\n            'PAY': 'EPAY',\n            'REP': 'Republicoin',\n        };\n        if (currency in substitutions)\n            return substitutions[currency];\n        return currency;\n    }\n\n    currencyId (commonCode) {\n        let substitutions = {\n            'AirCoin': 'AIR',\n            'ANICoin': 'ANI',\n            'AntsCoin': 'ANT',\n            'Autumncoin': 'ATM',\n            'BCH': 'BCC',\n            'Bitshares2': 'BTS',\n            'Discount': 'DCT',\n            'DarkGoldCoin': 'DGD',\n            'iCoin': 'ICN',\n            'LiZi': 'LIZI',\n            'LunarCoin': 'LUN',\n            'NavajoCoin': 'NAV',\n            'OMGame': 'OMG',\n            'EPAY': 'PAY',\n            'Republicoin': 'REP',\n        };\n        if (commonCode in substitutions)\n            return substitutions[commonCode];\n        return commonCode;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetInfo ();\n        let balances = response['return'];\n        let result = { 'info': balances };\n        let sides = { 'free': 'funds', 'total': 'funds_incl_orders' };\n        let keys = Object.keys (sides);\n        for (let i = 0; i < keys.length; i++) {\n            let key = keys[i];\n            let side = sides[key];\n            if (side in balances) {\n                let currencies = Object.keys (balances[side]);\n                for (let j = 0; j < currencies.length; j++) {\n                    let lowercase = currencies[j];\n                    let uppercase = lowercase.toUpperCase ();\n                    let currency = this.commonCurrencyCode (uppercase);\n                    let account = undefined;\n                    if (currency in result) {\n                        account = result[currency];\n                    } else {\n                        account = this.account ();\n                    }\n                    account[key] = balances[side][lowercase];\n                    if (account['total'] && account['free'])\n                        account['used'] = account['total'] - account['free'];\n                    result[currency] = account;\n                }\n            }\n        }\n        return this.parseBalance (result);\n    }\n\n    async createDepositAddress (currency, params = {}) {\n        let response = await this.fetchDepositAddress (currency, this.extend ({\n            'need_new': 1,\n        }, params));\n        return {\n            'currency': currency,\n            'address': response['address'],\n            'status': 'ok',\n            'info': response['info'],\n        };\n    }\n\n    async fetchDepositAddress (currency, params = {}) {\n        let currencyId = this.currencyId (currency);\n        let request = {\n            'coinName': currencyId,\n            'need_new': 0,\n        };\n        let response = await this.privatePostGetDepositAddress (this.extend (request, params));\n        let address = this.safeString (response['return'], 'address');\n        return {\n            'currency': currency,\n            'address': address,\n            'status': 'ok',\n            'info': response,\n        };\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostWithdrawCoinsToAddress (this.extend ({\n            'coinName': currency,\n            'amount': amount,\n            'address': address,\n        }, params));\n        return {\n            'info': response,\n            'id': undefined,\n        };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('success' in response) {\n            if (!response['success']) {\n                if (response['error'].indexOf ('Insufficient funds') >= 0) { // not enougTh is a typo inside Liqui's own API...\n                    throw new InsufficientFunds (this.id + ' ' + this.json (response));\n                } else if (response['error'] === 'Requests too often') {\n                    throw new DDoSProtection (this.id + ' ' + this.json (response));\n                } else if ((response['error'] === 'not available') || (response['error'] === 'external service unavailable')) {\n                    throw new DDoSProtection (this.id + ' ' + this.json (response));\n                } else {\n                    throw new ExchangeError (this.id + ' ' + this.json (response));\n                }\n            }\n        }\n        return response;\n    }\n\n}\n","\"use strict\";\n\n// ---------------------------------------------------------------------------\n\nconst acx = require ('./acx.js');\n\n// ---------------------------------------------------------------------------\n\nmodule.exports = class yunbi extends acx {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'yunbi',\n            'name': 'YUNBI',\n            'countries': 'CN',\n            'rateLimit': 1000,\n            'version': 'v2',\n            'has': {\n                'CORS': false,\n                'fetchTickers': true,\n                'fetchOHLCV': true,\n            },\n            'timeframes': {\n                '1m': '1',\n                '5m': '5',\n                '15m': '15',\n                '30m': '30',\n                '1h': '60',\n                '2h': '120',\n                '4h': '240',\n                '12h': '720',\n                '1d': '1440',\n                '3d': '4320',\n                '1w': '10080',\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/28570548-4d646c40-7147-11e7-9cf6-839b93e6d622.jpg',\n                'extension': '.json', // default extension appended to endpoint URLs\n                'api': 'https://yunbi.com',\n                'www': 'https://yunbi.com',\n                'doc': [\n                    'https://yunbi.com/documents/api/guide',\n                    'https://yunbi.com/swagger/',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'tickers',\n                        'tickers/{market}',\n                        'markets',\n                        'order_book',\n                        'k',\n                        'depth',\n                        'trades',\n                        'k_with_pending_trades',\n                        'timestamp',\n                        'addresses/{address}',\n                        'partners/orders/{id}/trades',\n                    ],\n                },\n                'private': {\n                    'get': [\n                        'deposits',\n                        'members/me',\n                        'deposit',\n                        'deposit_address',\n                        'order',\n                        'orders',\n                        'trades/my',\n                    ],\n                    'post': [\n                        'order/delete',\n                        'orders',\n                        'orders/multi',\n                        'orders/clear',\n                    ],\n                },\n            },\n        });\n    }\n}\n","'use strict';\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class zaif extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'zaif',\n            'name': 'Zaif',\n            'countries': 'JP',\n            'rateLimit': 2000,\n            'version': '1',\n            'has': {\n                'CORS': false,\n                'fetchOpenOrders': true,\n                'fetchClosedOrders': true,\n                'withdraw': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/27766927-39ca2ada-5eeb-11e7-972f-1b4199518ca6.jpg',\n                'api': 'https://api.zaif.jp',\n                'www': 'https://zaif.jp',\n                'doc': [\n                    'http://techbureau-api-document.readthedocs.io/ja/latest/index.html',\n                    'https://corp.zaif.jp/api-docs',\n                    'https://corp.zaif.jp/api-docs/api_links',\n                    'https://www.npmjs.com/package/zaif.jp',\n                    'https://github.com/you21979/node-zaif',\n                ],\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'depth/{pair}',\n                        'currencies/{pair}',\n                        'currencies/all',\n                        'currency_pairs/{pair}',\n                        'currency_pairs/all',\n                        'last_price/{pair}',\n                        'ticker/{pair}',\n                        'trades/{pair}',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'active_orders',\n                        'cancel_order',\n                        'deposit_history',\n                        'get_id_info',\n                        'get_info',\n                        'get_info2',\n                        'get_personal_info',\n                        'trade',\n                        'trade_history',\n                        'withdraw',\n                        'withdraw_history',\n                    ],\n                },\n                'ecapi': {\n                    'post': [\n                        'createInvoice',\n                        'getInvoice',\n                        'getInvoiceIdsByOrderNumber',\n                        'cancelInvoice',\n                    ],\n                },\n                'tlapi': {\n                    'post': [\n                        'get_positions',\n                        'position_history',\n                        'active_positions',\n                        'create_position',\n                        'change_position',\n                        'cancel_position',\n                    ],\n                },\n                'fapi': {\n                    'get': [\n                        'groups/{group_id}',\n                        'last_price/{group_id}/{pair}',\n                        'ticker/{group_id}/{pair}',\n                        'trades/{group_id}/{pair}',\n                        'depth/{group_id}/{pair}',\n                    ],\n                },\n            },\n        });\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetCurrencyPairsAll ();\n        let result = [];\n        for (let p = 0; p < markets.length; p++) {\n            let market = markets[p];\n            let id = market['currency_pair'];\n            let symbol = market['name'];\n            let [ base, quote ] = symbol.split ('/');\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'base': base,\n                'quote': quote,\n                'info': market,\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetInfo ();\n        let balances = response['return'];\n        let result = { 'info': balances };\n        let currencies = Object.keys (balances['funds']);\n        for (let c = 0; c < currencies.length; c++) {\n            let currency = currencies[c];\n            let balance = balances['funds'][currency];\n            let uppercase = currency.toUpperCase ();\n            let account = {\n                'free': balance,\n                'used': 0.0,\n                'total': balance,\n            };\n            if ('deposit' in balances) {\n                if (currency in balances['deposit']) {\n                    account['total'] = balances['deposit'][currency];\n                    account['used'] = account['total'] - account['free'];\n                }\n            }\n            result[uppercase] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let orderbook = await this.publicGetDepthPair (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        return this.parseOrderBook (orderbook);\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let ticker = await this.publicGetTickerPair (this.extend ({\n            'pair': this.marketId (symbol),\n        }, params));\n        let timestamp = this.milliseconds ();\n        let vwap = ticker['vwap'];\n        let baseVolume = ticker['volume'];\n        let quoteVolume = baseVolume * vwap;\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': ticker['high'],\n            'low': ticker['low'],\n            'bid': ticker['bid'],\n            'ask': ticker['ask'],\n            'vwap': vwap,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': ticker['last'],\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': baseVolume,\n            'quoteVolume': quoteVolume,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let side = (trade['trade_type'] === 'bid') ? 'buy' : 'sell';\n        let timestamp = trade['date'] * 1000;\n        let id = this.safeString (trade, 'id');\n        id = this.safeString (trade, 'tid', id);\n        if (!market)\n            market = this.markets_by_id[trade['currency_pair']];\n        return {\n            'id': id.toString (),\n            'info': trade,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': side,\n            'price': trade['price'],\n            'amount': trade['amount'],\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let response = await this.publicGetTradesPair (this.extend ({\n            'pair': market['id'],\n        }, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        if (type === 'market')\n            throw new ExchangeError (this.id + ' allows limit orders only');\n        let response = await this.privatePostTrade (this.extend ({\n            'currency_pair': this.marketId (symbol),\n            'action': (side === 'buy') ? 'bid' : 'ask',\n            'amount': amount,\n            'price': price,\n        }, params));\n        return {\n            'info': response,\n            'id': response['return']['order_id'].toString (),\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        return await this.privatePostCancelOrder (this.extend ({\n            'order_id': id,\n        }, params));\n    }\n\n    parseOrder (order, market = undefined) {\n        let side = (order['action'] === 'bid') ? 'buy' : 'sell';\n        let timestamp = parseInt (order['timestamp']) * 1000;\n        if (!market)\n            market = this.markets_by_id[order['currency_pair']];\n        let price = order['price'];\n        let amount = order['amount'];\n        return {\n            'id': order['id'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'status': 'open',\n            'symbol': market['symbol'],\n            'type': 'limit',\n            'side': side,\n            'price': price,\n            'cost': price * amount,\n            'amount': amount,\n            'filled': undefined,\n            'remaining': undefined,\n            'trades': undefined,\n            'fee': undefined,\n        };\n    }\n\n    parseOrders (orders, market = undefined, since = undefined, limit = undefined) {\n        let ids = Object.keys (orders);\n        let result = [];\n        for (let i = 0; i < ids.length; i++) {\n            let id = ids[i];\n            let order = orders[id];\n            let extended = this.extend (order, { 'id': id });\n            result.push (this.parseOrder (extended, market));\n        }\n        return this.filterBySinceLimit (result, since, limit);\n    }\n\n    async fetchOpenOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        let request = {\n            // 'is_token': false,\n            // 'is_token_both': false,\n        };\n        if (symbol) {\n            market = this.market (symbol);\n            request['currency_pair'] = market['id'];\n        }\n        let response = await this.privatePostActiveOrders (this.extend (request, params));\n        return this.parseOrders (response['return'], market, since, limit);\n    }\n\n    async fetchClosedOrders (symbol = undefined, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = undefined;\n        let request = {\n            // 'from': 0,\n            // 'count': 1000,\n            // 'from_id': 0,\n            // 'end_id': 1000,\n            // 'order': 'DESC',\n            // 'since': 1503821051,\n            // 'end': 1503821051,\n            // 'is_token': false,\n        };\n        if (symbol) {\n            market = this.market (symbol);\n            request['currency_pair'] = market['id'];\n        }\n        let response = await this.privatePostTradeHistory (this.extend (request, params));\n        return this.parseOrders (response['return'], market, since, limit);\n    }\n\n    async withdraw (currency, amount, address, tag = undefined, params = {}) {\n        await this.loadMarkets ();\n        if (currency === 'JPY')\n            throw new ExchangeError (this.id + ' does not allow ' + currency + ' withdrawals');\n        let result = await this.privatePostWithdraw (this.extend ({\n            'currency': currency,\n            'amount': amount,\n            'address': address,\n            // 'message': 'Hi!', // XEM only\n            // 'opt_fee': 0.003, // BTC and MONA only\n        }, params));\n        return {\n            'info': result,\n            'id': result['return']['txid'],\n            'fee': result['return']['fee'],\n        };\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'] + '/';\n        if (api === 'public') {\n            url += 'api/' + this.version + '/' + this.implodeParams (path, params);\n        } else if (api === 'fapi') {\n            url += 'fapi/' + this.version + '/' + this.implodeParams (path, params);\n        } else {\n            this.checkRequiredCredentials ();\n            if (api === 'ecapi') {\n                url += 'ecapi';\n            } else if (api === 'tlapi') {\n                url += 'tlapi';\n            } else {\n                url += 'tapi';\n            }\n            let nonce = this.nonce ();\n            body = this.urlencode (this.extend ({\n                'method': path,\n                'nonce': nonce,\n            }, params));\n            headers = {\n                'Content-Type': 'application/x-www-form-urlencoded',\n                'Key': this.apiKey,\n                'Sign': this.hmac (this.encode (body), this.encode (this.secret), 'sha512'),\n            };\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'api', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if ('error' in response)\n            throw new ExchangeError (this.id + ' ' + response['error']);\n        if ('success' in response)\n            if (!response['success'])\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n","\"use strict\";\n\n//  ---------------------------------------------------------------------------\n\nconst Exchange = require ('./base/Exchange');\nconst { ExchangeError } = require ('./base/errors');\n\n//  ---------------------------------------------------------------------------\n\nmodule.exports = class zb extends Exchange {\n\n    describe () {\n        return this.deepExtend (super.describe (), {\n            'id': 'zb',\n            'name': 'ZB',\n            'countries': 'CN',\n            'rateLimit': 1000,\n            'version': 'v1',\n            'has': {\n                'CORS': false,\n                'fetchOrder': true,\n            },\n            'urls': {\n                'logo': 'https://user-images.githubusercontent.com/1294454/32859187-cd5214f0-ca5e-11e7-967d-96568e2e2bd1.jpg',\n                'api': {\n                    'public': 'http://api.zb.com/data', // no https for public API\n                    'private': 'https://trade.zb.com/api',\n                },\n                'www': 'https://trade.zb.com/api',\n                'doc': 'https://www.zb.com/i/developer',\n                'fees': 'https://www.zb.com/i/rate',\n            },\n            'api': {\n                'public': {\n                    'get': [\n                        'markets',\n                        'ticker',\n                        'depth',\n                        'trades',\n                        'kline',\n                    ],\n                },\n                'private': {\n                    'post': [\n                        'order',\n                        'cancelOrder',\n                        'getOrder',\n                        'getOrders',\n                        'getOrdersNew',\n                        'getOrdersIgnoreTradeType',\n                        'getUnfinishedOrdersIgnoreTradeType',\n                        'getAccountInfo',\n                        'getUserAddress',\n                        'getWithdrawAddress',\n                        'getWithdrawRecord',\n                        'getChargeRecord',\n                        'getCnyWithdrawRecord',\n                        'getCnyChargeRecord',\n                        'withdraw',\n                    ],\n                },\n            },\n            'fees': {\n                'funding': {\n                    'withdraw': {\n                        'BTC': 0.0001,\n                        'BCH': 0.0006,\n                        'LTC': 0.005,\n                        'ETH': 0.01,\n                        'ETC': 0.01,\n                        'BTS': 3,\n                        'EOS': 1,\n                        'QTUM': 0.01,\n                        'HSR': 0.001,\n                        'XRP': 0.1,\n                        'USDT': '0.1%',\n                        'QCASH': 5,\n                        'DASH': 0.002,\n                        'BCD': 0,\n                        'UBTC': 0,\n                        'SBTC': 0,\n                        'INK': 20,\n                        'TV': 0.1,\n                        'BTH': 0,\n                        'BCX': 0,\n                        'LBTC': 0,\n                        'CHAT': 20,\n                        'bitCNY': 20,\n                        'HLC': 20,\n                        'BTP': 0,\n                        'BCW': 0,\n                    },\n                },\n                'trading': {\n                },\n            },\n        });\n    }\n\n    getTradingFeeFromBaseQuote (base, quote) {\n        // base: quote\n        let fees = {\n            'BTC': { 'USDT': 0.0 },\n            'BCH': { 'BTC': 0.001, 'USDT': 0.001 },\n            'LTC': { 'BTC': 0.001, 'USDT': 0.0 },\n            'ETH': { 'BTC': 0.001, 'USDT': 0.0 },\n            'ETC': { 'BTC': 0.001, 'USDT': 0.0 },\n            'BTS': { 'BTC': 0.001, 'USDT': 0.001 },\n            'EOS': { 'BTC': 0.001, 'USDT': 0.001 },\n            'HSR': { 'BTC': 0.001, 'USDT': 0.001 },\n            'QTUM': { 'BTC': 0.001, 'USDT': 0.001 },\n            'USDT': { 'BTC': 0.0 },\n        };\n        if (base in fees) {\n            let quoteFees = fees[base];\n            if (quote in quoteFees)\n                return quoteFees[quote];\n        }\n        return undefined;\n    }\n\n    async fetchMarkets () {\n        let markets = await this.publicGetMarkets ();\n        let keys = Object.keys (markets);\n        let result = [];\n        for (let i = 0; i < keys.length; i++) {\n            let id = keys[i];\n            let market = markets[id];\n            let [ baseId, quoteId ] = id.split ('_');\n            let base = this.commonCurrencyCode (baseId.toUpperCase ());\n            let quote = this.commonCurrencyCode (quoteId.toUpperCase ());\n            let symbol = base + '/' + quote;\n            let fee = this.getTradingFeeFromBaseQuote (base, quote);\n            let precision = {\n                'amount': market['amountScale'],\n                'price': market['priceScale'],\n            };\n            let lot = Math.pow (10, -precision['amount']);\n            result.push ({\n                'id': id,\n                'symbol': symbol,\n                'baseId': baseId,\n                'quoteId': quoteId,\n                'base': base,\n                'quote': quote,\n                'info': market,\n                'maker': fee,\n                'taker': fee,\n                'lot': lot,\n                'active': true,\n                'precision': precision,\n                'limits': {\n                    'amount': {\n                        'min': lot,\n                        'max': undefined,\n                    },\n                    'price': {\n                        'min': Math.pow (10, -precision['price']),\n                        'max': undefined,\n                    },\n                    'cost': {\n                        'min': 0,\n                        'max': undefined,\n                    },\n                },\n            });\n        }\n        return result;\n    }\n\n    async fetchBalance (params = {}) {\n        await this.loadMarkets ();\n        let response = await this.privatePostGetAccountInfo ();\n        let balances = response['result'];\n        let result = { 'info': balances };\n        let currencies = Object.keys (this.currencies);\n        for (let i = 0; i < currencies.length; i++) {\n            let currency = currencies[i];\n            let account = this.account ();\n            if (currency in balances['balance'])\n                account['free'] = parseFloat (balances['balance'][currency]['amount']);\n            if (currency in balances['frozen'])\n                account['used'] = parseFloat (balances['frozen'][currency]['amount']);\n            account['total'] = this.sum (account['free'], account['used']);\n            result[currency] = account;\n        }\n        return this.parseBalance (result);\n    }\n\n    getMarketFieldName () {\n        return 'market';\n    }\n\n    async fetchOrderBook (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let marketFieldName = this.getMarketFieldName ();\n        let request = {};\n        request[marketFieldName] = market['id'];\n        let orderbook = await this.publicGetDepth (this.extend (request, params));\n        let timestamp = this.milliseconds ();\n        let bids = undefined;\n        let asks = undefined;\n        if ('bids' in orderbook)\n            bids = orderbook['bids'];\n        if ('asks' in orderbook)\n            asks = orderbook['asks'];\n        let result = {\n            'bids': bids,\n            'asks': asks,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n        };\n        if (result['bids'])\n            result['bids'] = this.sortBy (result['bids'], 0, true);\n        if (result['asks'])\n            result['asks'] = this.sortBy (result['asks'], 0);\n        return result;\n    }\n\n    async fetchTicker (symbol, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let marketFieldName = this.getMarketFieldName ();\n        let request = {};\n        request[marketFieldName] = market['id'];\n        let response = await this.publicGetTicker (this.extend (request, params));\n        let ticker = response['ticker'];\n        let timestamp = this.milliseconds ();\n        return {\n            'symbol': symbol,\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'high': parseFloat (ticker['high']),\n            'low': parseFloat (ticker['low']),\n            'bid': parseFloat (ticker['buy']),\n            'ask': parseFloat (ticker['sell']),\n            'vwap': undefined,\n            'open': undefined,\n            'close': undefined,\n            'first': undefined,\n            'last': parseFloat (ticker['last']),\n            'change': undefined,\n            'percentage': undefined,\n            'average': undefined,\n            'baseVolume': parseFloat (ticker['vol']),\n            'quoteVolume': undefined,\n            'info': ticker,\n        };\n    }\n\n    parseTrade (trade, market = undefined) {\n        let timestamp = trade['date'] * 1000;\n        let side = (trade['trade_type'] == 'bid') ? 'buy' : 'sell';\n        return {\n            'info': trade,\n            'id': trade['tid'].toString (),\n            'timestamp': timestamp,\n            'datetime': this.iso8601 (timestamp),\n            'symbol': market['symbol'],\n            'type': undefined,\n            'side': side,\n            'price': parseFloat (trade['price']),\n            'amount': parseFloat (trade['amount']),\n        };\n    }\n\n    async fetchTrades (symbol, since = undefined, limit = undefined, params = {}) {\n        await this.loadMarkets ();\n        let market = this.market (symbol);\n        let marketFieldName = this.getMarketFieldName ();\n        let request = {};\n        request[marketFieldName] = market['id'];\n        let response = await this.publicGetTrades (this.extend (request, params));\n        return this.parseTrades (response, market, since, limit);\n    }\n\n    async createOrder (symbol, type, side, amount, price = undefined, params = {}) {\n        await this.loadMarkets ();\n        let paramString = '&price=' + price.toString ();\n        paramString += '&amount=' + amount.toString ();\n        let tradeType = (side == 'buy') ? '1' : '0';\n        paramString += '&tradeType=' + tradeType;\n        paramString += '&currency=' + this.marketId (symbol);\n        let response = await this.privatePostOrder (paramString);\n        return {\n            'info': response,\n            'id': response['id'],\n        };\n    }\n\n    async cancelOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let paramString = '&id=' + id.toString ();\n        if ('currency' in params)\n            paramString += '&currency=' + params['currency'];\n        return await this.privatePostCancelOrder (paramString);\n    }\n\n    async fetchOrder (id, symbol = undefined, params = {}) {\n        await this.loadMarkets ();\n        let paramString = '&id=' + id.toString ();\n        if ('currency' in params)\n            paramString += '&currency=' + params['currency'];\n        return await this.privatePostGetOrder (paramString);\n    }\n\n    nonce () {\n        return this.milliseconds ();\n    }\n\n    sign (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let url = this.urls['api'][api];\n        if (api == 'public') {\n            url += '/' + this.version + '/' + path;\n            if (Object.keys (params).length)\n                url += '?' + this.urlencode (params);\n        } else {\n            this.checkRequiredCredentials ();\n            let nonce = this.nonce ();\n            let auth = 'accesskey=' + this.apiKey;\n            auth += '&method=' + path;\n            let secret = this.hash (this.encode (this.secret), 'sha1');\n            let signature = this.hmac (this.encode (auth), this.encode (secret), 'md5');\n            let suffix = 'sign=' + signature + '&reqTime=' + nonce.toString ();\n            url += '/' + path + '?' + auth + '&' + suffix;\n        }\n        return { 'url': url, 'method': method, 'body': body, 'headers': headers };\n    }\n\n    async request (path, api = 'public', method = 'GET', params = {}, headers = undefined, body = undefined) {\n        let response = await this.fetch2 (path, api, method, params, headers, body);\n        if (api == 'private')\n            if ('code' in response)\n                throw new ExchangeError (this.id + ' ' + this.json (response));\n        return response;\n    }\n}\n",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./enc-base64\"), require(\"./md5\"), require(\"./evpkdf\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./enc-base64\", \"./md5\", \"./evpkdf\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var BlockCipher = C_lib.BlockCipher;\n\t    var C_algo = C.algo;\n\n\t    // Lookup tables\n\t    var SBOX = [];\n\t    var INV_SBOX = [];\n\t    var SUB_MIX_0 = [];\n\t    var SUB_MIX_1 = [];\n\t    var SUB_MIX_2 = [];\n\t    var SUB_MIX_3 = [];\n\t    var INV_SUB_MIX_0 = [];\n\t    var INV_SUB_MIX_1 = [];\n\t    var INV_SUB_MIX_2 = [];\n\t    var INV_SUB_MIX_3 = [];\n\n\t    // Compute lookup tables\n\t    (function () {\n\t        // Compute double table\n\t        var d = [];\n\t        for (var i = 0; i < 256; i++) {\n\t            if (i < 128) {\n\t                d[i] = i << 1;\n\t            } else {\n\t                d[i] = (i << 1) ^ 0x11b;\n\t            }\n\t        }\n\n\t        // Walk GF(2^8)\n\t        var x = 0;\n\t        var xi = 0;\n\t        for (var i = 0; i < 256; i++) {\n\t            // Compute sbox\n\t            var sx = xi ^ (xi << 1) ^ (xi << 2) ^ (xi << 3) ^ (xi << 4);\n\t            sx = (sx >>> 8) ^ (sx & 0xff) ^ 0x63;\n\t            SBOX[x] = sx;\n\t            INV_SBOX[sx] = x;\n\n\t            // Compute multiplication\n\t            var x2 = d[x];\n\t            var x4 = d[x2];\n\t            var x8 = d[x4];\n\n\t            // Compute sub bytes, mix columns tables\n\t            var t = (d[sx] * 0x101) ^ (sx * 0x1010100);\n\t            SUB_MIX_0[x] = (t << 24) | (t >>> 8);\n\t            SUB_MIX_1[x] = (t << 16) | (t >>> 16);\n\t            SUB_MIX_2[x] = (t << 8)  | (t >>> 24);\n\t            SUB_MIX_3[x] = t;\n\n\t            // Compute inv sub bytes, inv mix columns tables\n\t            var t = (x8 * 0x1010101) ^ (x4 * 0x10001) ^ (x2 * 0x101) ^ (x * 0x1010100);\n\t            INV_SUB_MIX_0[sx] = (t << 24) | (t >>> 8);\n\t            INV_SUB_MIX_1[sx] = (t << 16) | (t >>> 16);\n\t            INV_SUB_MIX_2[sx] = (t << 8)  | (t >>> 24);\n\t            INV_SUB_MIX_3[sx] = t;\n\n\t            // Compute next counter\n\t            if (!x) {\n\t                x = xi = 1;\n\t            } else {\n\t                x = x2 ^ d[d[d[x8 ^ x2]]];\n\t                xi ^= d[d[xi]];\n\t            }\n\t        }\n\t    }());\n\n\t    // Precomputed Rcon lookup\n\t    var RCON = [0x00, 0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 0x1b, 0x36];\n\n\t    /**\n\t     * AES block cipher algorithm.\n\t     */\n\t    var AES = C_algo.AES = BlockCipher.extend({\n\t        _doReset: function () {\n\t            // Skip reset of nRounds has been set before and key did not change\n\t            if (this._nRounds && this._keyPriorReset === this._key) {\n\t                return;\n\t            }\n\n\t            // Shortcuts\n\t            var key = this._keyPriorReset = this._key;\n\t            var keyWords = key.words;\n\t            var keySize = key.sigBytes / 4;\n\n\t            // Compute number of rounds\n\t            var nRounds = this._nRounds = keySize + 6;\n\n\t            // Compute number of key schedule rows\n\t            var ksRows = (nRounds + 1) * 4;\n\n\t            // Compute key schedule\n\t            var keySchedule = this._keySchedule = [];\n\t            for (var ksRow = 0; ksRow < ksRows; ksRow++) {\n\t                if (ksRow < keySize) {\n\t                    keySchedule[ksRow] = keyWords[ksRow];\n\t                } else {\n\t                    var t = keySchedule[ksRow - 1];\n\n\t                    if (!(ksRow % keySize)) {\n\t                        // Rot word\n\t                        t = (t << 8) | (t >>> 24);\n\n\t                        // Sub word\n\t                        t = (SBOX[t >>> 24] << 24) | (SBOX[(t >>> 16) & 0xff] << 16) | (SBOX[(t >>> 8) & 0xff] << 8) | SBOX[t & 0xff];\n\n\t                        // Mix Rcon\n\t                        t ^= RCON[(ksRow / keySize) | 0] << 24;\n\t                    } else if (keySize > 6 && ksRow % keySize == 4) {\n\t                        // Sub word\n\t                        t = (SBOX[t >>> 24] << 24) | (SBOX[(t >>> 16) & 0xff] << 16) | (SBOX[(t >>> 8) & 0xff] << 8) | SBOX[t & 0xff];\n\t                    }\n\n\t                    keySchedule[ksRow] = keySchedule[ksRow - keySize] ^ t;\n\t                }\n\t            }\n\n\t            // Compute inv key schedule\n\t            var invKeySchedule = this._invKeySchedule = [];\n\t            for (var invKsRow = 0; invKsRow < ksRows; invKsRow++) {\n\t                var ksRow = ksRows - invKsRow;\n\n\t                if (invKsRow % 4) {\n\t                    var t = keySchedule[ksRow];\n\t                } else {\n\t                    var t = keySchedule[ksRow - 4];\n\t                }\n\n\t                if (invKsRow < 4 || ksRow <= 4) {\n\t                    invKeySchedule[invKsRow] = t;\n\t                } else {\n\t                    invKeySchedule[invKsRow] = INV_SUB_MIX_0[SBOX[t >>> 24]] ^ INV_SUB_MIX_1[SBOX[(t >>> 16) & 0xff]] ^\n\t                                               INV_SUB_MIX_2[SBOX[(t >>> 8) & 0xff]] ^ INV_SUB_MIX_3[SBOX[t & 0xff]];\n\t                }\n\t            }\n\t        },\n\n\t        encryptBlock: function (M, offset) {\n\t            this._doCryptBlock(M, offset, this._keySchedule, SUB_MIX_0, SUB_MIX_1, SUB_MIX_2, SUB_MIX_3, SBOX);\n\t        },\n\n\t        decryptBlock: function (M, offset) {\n\t            // Swap 2nd and 4th rows\n\t            var t = M[offset + 1];\n\t            M[offset + 1] = M[offset + 3];\n\t            M[offset + 3] = t;\n\n\t            this._doCryptBlock(M, offset, this._invKeySchedule, INV_SUB_MIX_0, INV_SUB_MIX_1, INV_SUB_MIX_2, INV_SUB_MIX_3, INV_SBOX);\n\n\t            // Inv swap 2nd and 4th rows\n\t            var t = M[offset + 1];\n\t            M[offset + 1] = M[offset + 3];\n\t            M[offset + 3] = t;\n\t        },\n\n\t        _doCryptBlock: function (M, offset, keySchedule, SUB_MIX_0, SUB_MIX_1, SUB_MIX_2, SUB_MIX_3, SBOX) {\n\t            // Shortcut\n\t            var nRounds = this._nRounds;\n\n\t            // Get input, add round key\n\t            var s0 = M[offset]     ^ keySchedule[0];\n\t            var s1 = M[offset + 1] ^ keySchedule[1];\n\t            var s2 = M[offset + 2] ^ keySchedule[2];\n\t            var s3 = M[offset + 3] ^ keySchedule[3];\n\n\t            // Key schedule row counter\n\t            var ksRow = 4;\n\n\t            // Rounds\n\t            for (var round = 1; round < nRounds; round++) {\n\t                // Shift rows, sub bytes, mix columns, add round key\n\t                var t0 = SUB_MIX_0[s0 >>> 24] ^ SUB_MIX_1[(s1 >>> 16) & 0xff] ^ SUB_MIX_2[(s2 >>> 8) & 0xff] ^ SUB_MIX_3[s3 & 0xff] ^ keySchedule[ksRow++];\n\t                var t1 = SUB_MIX_0[s1 >>> 24] ^ SUB_MIX_1[(s2 >>> 16) & 0xff] ^ SUB_MIX_2[(s3 >>> 8) & 0xff] ^ SUB_MIX_3[s0 & 0xff] ^ keySchedule[ksRow++];\n\t                var t2 = SUB_MIX_0[s2 >>> 24] ^ SUB_MIX_1[(s3 >>> 16) & 0xff] ^ SUB_MIX_2[(s0 >>> 8) & 0xff] ^ SUB_MIX_3[s1 & 0xff] ^ keySchedule[ksRow++];\n\t                var t3 = SUB_MIX_0[s3 >>> 24] ^ SUB_MIX_1[(s0 >>> 16) & 0xff] ^ SUB_MIX_2[(s1 >>> 8) & 0xff] ^ SUB_MIX_3[s2 & 0xff] ^ keySchedule[ksRow++];\n\n\t                // Update state\n\t                s0 = t0;\n\t                s1 = t1;\n\t                s2 = t2;\n\t                s3 = t3;\n\t            }\n\n\t            // Shift rows, sub bytes, add round key\n\t            var t0 = ((SBOX[s0 >>> 24] << 24) | (SBOX[(s1 >>> 16) & 0xff] << 16) | (SBOX[(s2 >>> 8) & 0xff] << 8) | SBOX[s3 & 0xff]) ^ keySchedule[ksRow++];\n\t            var t1 = ((SBOX[s1 >>> 24] << 24) | (SBOX[(s2 >>> 16) & 0xff] << 16) | (SBOX[(s3 >>> 8) & 0xff] << 8) | SBOX[s0 & 0xff]) ^ keySchedule[ksRow++];\n\t            var t2 = ((SBOX[s2 >>> 24] << 24) | (SBOX[(s3 >>> 16) & 0xff] << 16) | (SBOX[(s0 >>> 8) & 0xff] << 8) | SBOX[s1 & 0xff]) ^ keySchedule[ksRow++];\n\t            var t3 = ((SBOX[s3 >>> 24] << 24) | (SBOX[(s0 >>> 16) & 0xff] << 16) | (SBOX[(s1 >>> 8) & 0xff] << 8) | SBOX[s2 & 0xff]) ^ keySchedule[ksRow++];\n\n\t            // Set output\n\t            M[offset]     = t0;\n\t            M[offset + 1] = t1;\n\t            M[offset + 2] = t2;\n\t            M[offset + 3] = t3;\n\t        },\n\n\t        keySize: 256/32\n\t    });\n\n\t    /**\n\t     * Shortcut functions to the cipher's object interface.\n\t     *\n\t     * @example\n\t     *\n\t     *     var ciphertext = CryptoJS.AES.encrypt(message, key, cfg);\n\t     *     var plaintext  = CryptoJS.AES.decrypt(ciphertext, key, cfg);\n\t     */\n\t    C.AES = BlockCipher._createHelper(AES);\n\t}());\n\n\n\treturn CryptoJS.AES;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./evpkdf\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./evpkdf\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * Cipher core components.\n\t */\n\tCryptoJS.lib.Cipher || (function (undefined) {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var Base = C_lib.Base;\n\t    var WordArray = C_lib.WordArray;\n\t    var BufferedBlockAlgorithm = C_lib.BufferedBlockAlgorithm;\n\t    var C_enc = C.enc;\n\t    var Utf8 = C_enc.Utf8;\n\t    var Base64 = C_enc.Base64;\n\t    var C_algo = C.algo;\n\t    var EvpKDF = C_algo.EvpKDF;\n\n\t    /**\n\t     * Abstract base cipher template.\n\t     *\n\t     * @property {number} keySize This cipher's key size. Default: 4 (128 bits)\n\t     * @property {number} ivSize This cipher's IV size. Default: 4 (128 bits)\n\t     * @property {number} _ENC_XFORM_MODE A constant representing encryption mode.\n\t     * @property {number} _DEC_XFORM_MODE A constant representing decryption mode.\n\t     */\n\t    var Cipher = C_lib.Cipher = BufferedBlockAlgorithm.extend({\n\t        /**\n\t         * Configuration options.\n\t         *\n\t         * @property {WordArray} iv The IV to use for this operation.\n\t         */\n\t        cfg: Base.extend(),\n\n\t        /**\n\t         * Creates this cipher in encryption mode.\n\t         *\n\t         * @param {WordArray} key The key.\n\t         * @param {Object} cfg (Optional) The configuration options to use for this operation.\n\t         *\n\t         * @return {Cipher} A cipher instance.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var cipher = CryptoJS.algo.AES.createEncryptor(keyWordArray, { iv: ivWordArray });\n\t         */\n\t        createEncryptor: function (key, cfg) {\n\t            return this.create(this._ENC_XFORM_MODE, key, cfg);\n\t        },\n\n\t        /**\n\t         * Creates this cipher in decryption mode.\n\t         *\n\t         * @param {WordArray} key The key.\n\t         * @param {Object} cfg (Optional) The configuration options to use for this operation.\n\t         *\n\t         * @return {Cipher} A cipher instance.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var cipher = CryptoJS.algo.AES.createDecryptor(keyWordArray, { iv: ivWordArray });\n\t         */\n\t        createDecryptor: function (key, cfg) {\n\t            return this.create(this._DEC_XFORM_MODE, key, cfg);\n\t        },\n\n\t        /**\n\t         * Initializes a newly created cipher.\n\t         *\n\t         * @param {number} xformMode Either the encryption or decryption transormation mode constant.\n\t         * @param {WordArray} key The key.\n\t         * @param {Object} cfg (Optional) The configuration options to use for this operation.\n\t         *\n\t         * @example\n\t         *\n\t         *     var cipher = CryptoJS.algo.AES.create(CryptoJS.algo.AES._ENC_XFORM_MODE, keyWordArray, { iv: ivWordArray });\n\t         */\n\t        init: function (xformMode, key, cfg) {\n\t            // Apply config defaults\n\t            this.cfg = this.cfg.extend(cfg);\n\n\t            // Store transform mode and key\n\t            this._xformMode = xformMode;\n\t            this._key = key;\n\n\t            // Set initial values\n\t            this.reset();\n\t        },\n\n\t        /**\n\t         * Resets this cipher to its initial state.\n\t         *\n\t         * @example\n\t         *\n\t         *     cipher.reset();\n\t         */\n\t        reset: function () {\n\t            // Reset data buffer\n\t            BufferedBlockAlgorithm.reset.call(this);\n\n\t            // Perform concrete-cipher logic\n\t            this._doReset();\n\t        },\n\n\t        /**\n\t         * Adds data to be encrypted or decrypted.\n\t         *\n\t         * @param {WordArray|string} dataUpdate The data to encrypt or decrypt.\n\t         *\n\t         * @return {WordArray} The data after processing.\n\t         *\n\t         * @example\n\t         *\n\t         *     var encrypted = cipher.process('data');\n\t         *     var encrypted = cipher.process(wordArray);\n\t         */\n\t        process: function (dataUpdate) {\n\t            // Append\n\t            this._append(dataUpdate);\n\n\t            // Process available blocks\n\t            return this._process();\n\t        },\n\n\t        /**\n\t         * Finalizes the encryption or decryption process.\n\t         * Note that the finalize operation is effectively a destructive, read-once operation.\n\t         *\n\t         * @param {WordArray|string} dataUpdate The final data to encrypt or decrypt.\n\t         *\n\t         * @return {WordArray} The data after final processing.\n\t         *\n\t         * @example\n\t         *\n\t         *     var encrypted = cipher.finalize();\n\t         *     var encrypted = cipher.finalize('data');\n\t         *     var encrypted = cipher.finalize(wordArray);\n\t         */\n\t        finalize: function (dataUpdate) {\n\t            // Final data update\n\t            if (dataUpdate) {\n\t                this._append(dataUpdate);\n\t            }\n\n\t            // Perform concrete-cipher logic\n\t            var finalProcessedData = this._doFinalize();\n\n\t            return finalProcessedData;\n\t        },\n\n\t        keySize: 128/32,\n\n\t        ivSize: 128/32,\n\n\t        _ENC_XFORM_MODE: 1,\n\n\t        _DEC_XFORM_MODE: 2,\n\n\t        /**\n\t         * Creates shortcut functions to a cipher's object interface.\n\t         *\n\t         * @param {Cipher} cipher The cipher to create a helper for.\n\t         *\n\t         * @return {Object} An object with encrypt and decrypt shortcut functions.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var AES = CryptoJS.lib.Cipher._createHelper(CryptoJS.algo.AES);\n\t         */\n\t        _createHelper: (function () {\n\t            function selectCipherStrategy(key) {\n\t                if (typeof key == 'string') {\n\t                    return PasswordBasedCipher;\n\t                } else {\n\t                    return SerializableCipher;\n\t                }\n\t            }\n\n\t            return function (cipher) {\n\t                return {\n\t                    encrypt: function (message, key, cfg) {\n\t                        return selectCipherStrategy(key).encrypt(cipher, message, key, cfg);\n\t                    },\n\n\t                    decrypt: function (ciphertext, key, cfg) {\n\t                        return selectCipherStrategy(key).decrypt(cipher, ciphertext, key, cfg);\n\t                    }\n\t                };\n\t            };\n\t        }())\n\t    });\n\n\t    /**\n\t     * Abstract base stream cipher template.\n\t     *\n\t     * @property {number} blockSize The number of 32-bit words this cipher operates on. Default: 1 (32 bits)\n\t     */\n\t    var StreamCipher = C_lib.StreamCipher = Cipher.extend({\n\t        _doFinalize: function () {\n\t            // Process partial blocks\n\t            var finalProcessedBlocks = this._process(!!'flush');\n\n\t            return finalProcessedBlocks;\n\t        },\n\n\t        blockSize: 1\n\t    });\n\n\t    /**\n\t     * Mode namespace.\n\t     */\n\t    var C_mode = C.mode = {};\n\n\t    /**\n\t     * Abstract base block cipher mode template.\n\t     */\n\t    var BlockCipherMode = C_lib.BlockCipherMode = Base.extend({\n\t        /**\n\t         * Creates this mode for encryption.\n\t         *\n\t         * @param {Cipher} cipher A block cipher instance.\n\t         * @param {Array} iv The IV words.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var mode = CryptoJS.mode.CBC.createEncryptor(cipher, iv.words);\n\t         */\n\t        createEncryptor: function (cipher, iv) {\n\t            return this.Encryptor.create(cipher, iv);\n\t        },\n\n\t        /**\n\t         * Creates this mode for decryption.\n\t         *\n\t         * @param {Cipher} cipher A block cipher instance.\n\t         * @param {Array} iv The IV words.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var mode = CryptoJS.mode.CBC.createDecryptor(cipher, iv.words);\n\t         */\n\t        createDecryptor: function (cipher, iv) {\n\t            return this.Decryptor.create(cipher, iv);\n\t        },\n\n\t        /**\n\t         * Initializes a newly created mode.\n\t         *\n\t         * @param {Cipher} cipher A block cipher instance.\n\t         * @param {Array} iv The IV words.\n\t         *\n\t         * @example\n\t         *\n\t         *     var mode = CryptoJS.mode.CBC.Encryptor.create(cipher, iv.words);\n\t         */\n\t        init: function (cipher, iv) {\n\t            this._cipher = cipher;\n\t            this._iv = iv;\n\t        }\n\t    });\n\n\t    /**\n\t     * Cipher Block Chaining mode.\n\t     */\n\t    var CBC = C_mode.CBC = (function () {\n\t        /**\n\t         * Abstract base CBC mode.\n\t         */\n\t        var CBC = BlockCipherMode.extend();\n\n\t        /**\n\t         * CBC encryptor.\n\t         */\n\t        CBC.Encryptor = CBC.extend({\n\t            /**\n\t             * Processes the data block at offset.\n\t             *\n\t             * @param {Array} words The data words to operate on.\n\t             * @param {number} offset The offset where the block starts.\n\t             *\n\t             * @example\n\t             *\n\t             *     mode.processBlock(data.words, offset);\n\t             */\n\t            processBlock: function (words, offset) {\n\t                // Shortcuts\n\t                var cipher = this._cipher;\n\t                var blockSize = cipher.blockSize;\n\n\t                // XOR and encrypt\n\t                xorBlock.call(this, words, offset, blockSize);\n\t                cipher.encryptBlock(words, offset);\n\n\t                // Remember this block to use with next block\n\t                this._prevBlock = words.slice(offset, offset + blockSize);\n\t            }\n\t        });\n\n\t        /**\n\t         * CBC decryptor.\n\t         */\n\t        CBC.Decryptor = CBC.extend({\n\t            /**\n\t             * Processes the data block at offset.\n\t             *\n\t             * @param {Array} words The data words to operate on.\n\t             * @param {number} offset The offset where the block starts.\n\t             *\n\t             * @example\n\t             *\n\t             *     mode.processBlock(data.words, offset);\n\t             */\n\t            processBlock: function (words, offset) {\n\t                // Shortcuts\n\t                var cipher = this._cipher;\n\t                var blockSize = cipher.blockSize;\n\n\t                // Remember this block to use with next block\n\t                var thisBlock = words.slice(offset, offset + blockSize);\n\n\t                // Decrypt and XOR\n\t                cipher.decryptBlock(words, offset);\n\t                xorBlock.call(this, words, offset, blockSize);\n\n\t                // This block becomes the previous block\n\t                this._prevBlock = thisBlock;\n\t            }\n\t        });\n\n\t        function xorBlock(words, offset, blockSize) {\n\t            // Shortcut\n\t            var iv = this._iv;\n\n\t            // Choose mixing block\n\t            if (iv) {\n\t                var block = iv;\n\n\t                // Remove IV for subsequent blocks\n\t                this._iv = undefined;\n\t            } else {\n\t                var block = this._prevBlock;\n\t            }\n\n\t            // XOR blocks\n\t            for (var i = 0; i < blockSize; i++) {\n\t                words[offset + i] ^= block[i];\n\t            }\n\t        }\n\n\t        return CBC;\n\t    }());\n\n\t    /**\n\t     * Padding namespace.\n\t     */\n\t    var C_pad = C.pad = {};\n\n\t    /**\n\t     * PKCS #5/7 padding strategy.\n\t     */\n\t    var Pkcs7 = C_pad.Pkcs7 = {\n\t        /**\n\t         * Pads data using the algorithm defined in PKCS #5/7.\n\t         *\n\t         * @param {WordArray} data The data to pad.\n\t         * @param {number} blockSize The multiple that the data should be padded to.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     CryptoJS.pad.Pkcs7.pad(wordArray, 4);\n\t         */\n\t        pad: function (data, blockSize) {\n\t            // Shortcut\n\t            var blockSizeBytes = blockSize * 4;\n\n\t            // Count padding bytes\n\t            var nPaddingBytes = blockSizeBytes - data.sigBytes % blockSizeBytes;\n\n\t            // Create padding word\n\t            var paddingWord = (nPaddingBytes << 24) | (nPaddingBytes << 16) | (nPaddingBytes << 8) | nPaddingBytes;\n\n\t            // Create padding\n\t            var paddingWords = [];\n\t            for (var i = 0; i < nPaddingBytes; i += 4) {\n\t                paddingWords.push(paddingWord);\n\t            }\n\t            var padding = WordArray.create(paddingWords, nPaddingBytes);\n\n\t            // Add padding\n\t            data.concat(padding);\n\t        },\n\n\t        /**\n\t         * Unpads data that had been padded using the algorithm defined in PKCS #5/7.\n\t         *\n\t         * @param {WordArray} data The data to unpad.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     CryptoJS.pad.Pkcs7.unpad(wordArray);\n\t         */\n\t        unpad: function (data) {\n\t            // Get number of padding bytes from last byte\n\t            var nPaddingBytes = data.words[(data.sigBytes - 1) >>> 2] & 0xff;\n\n\t            // Remove padding\n\t            data.sigBytes -= nPaddingBytes;\n\t        }\n\t    };\n\n\t    /**\n\t     * Abstract base block cipher template.\n\t     *\n\t     * @property {number} blockSize The number of 32-bit words this cipher operates on. Default: 4 (128 bits)\n\t     */\n\t    var BlockCipher = C_lib.BlockCipher = Cipher.extend({\n\t        /**\n\t         * Configuration options.\n\t         *\n\t         * @property {Mode} mode The block mode to use. Default: CBC\n\t         * @property {Padding} padding The padding strategy to use. Default: Pkcs7\n\t         */\n\t        cfg: Cipher.cfg.extend({\n\t            mode: CBC,\n\t            padding: Pkcs7\n\t        }),\n\n\t        reset: function () {\n\t            // Reset cipher\n\t            Cipher.reset.call(this);\n\n\t            // Shortcuts\n\t            var cfg = this.cfg;\n\t            var iv = cfg.iv;\n\t            var mode = cfg.mode;\n\n\t            // Reset block mode\n\t            if (this._xformMode == this._ENC_XFORM_MODE) {\n\t                var modeCreator = mode.createEncryptor;\n\t            } else /* if (this._xformMode == this._DEC_XFORM_MODE) */ {\n\t                var modeCreator = mode.createDecryptor;\n\t                // Keep at least one block in the buffer for unpadding\n\t                this._minBufferSize = 1;\n\t            }\n\n\t            if (this._mode && this._mode.__creator == modeCreator) {\n\t                this._mode.init(this, iv && iv.words);\n\t            } else {\n\t                this._mode = modeCreator.call(mode, this, iv && iv.words);\n\t                this._mode.__creator = modeCreator;\n\t            }\n\t        },\n\n\t        _doProcessBlock: function (words, offset) {\n\t            this._mode.processBlock(words, offset);\n\t        },\n\n\t        _doFinalize: function () {\n\t            // Shortcut\n\t            var padding = this.cfg.padding;\n\n\t            // Finalize\n\t            if (this._xformMode == this._ENC_XFORM_MODE) {\n\t                // Pad data\n\t                padding.pad(this._data, this.blockSize);\n\n\t                // Process final blocks\n\t                var finalProcessedBlocks = this._process(!!'flush');\n\t            } else /* if (this._xformMode == this._DEC_XFORM_MODE) */ {\n\t                // Process final blocks\n\t                var finalProcessedBlocks = this._process(!!'flush');\n\n\t                // Unpad data\n\t                padding.unpad(finalProcessedBlocks);\n\t            }\n\n\t            return finalProcessedBlocks;\n\t        },\n\n\t        blockSize: 128/32\n\t    });\n\n\t    /**\n\t     * A collection of cipher parameters.\n\t     *\n\t     * @property {WordArray} ciphertext The raw ciphertext.\n\t     * @property {WordArray} key The key to this ciphertext.\n\t     * @property {WordArray} iv The IV used in the ciphering operation.\n\t     * @property {WordArray} salt The salt used with a key derivation function.\n\t     * @property {Cipher} algorithm The cipher algorithm.\n\t     * @property {Mode} mode The block mode used in the ciphering operation.\n\t     * @property {Padding} padding The padding scheme used in the ciphering operation.\n\t     * @property {number} blockSize The block size of the cipher.\n\t     * @property {Format} formatter The default formatting strategy to convert this cipher params object to a string.\n\t     */\n\t    var CipherParams = C_lib.CipherParams = Base.extend({\n\t        /**\n\t         * Initializes a newly created cipher params object.\n\t         *\n\t         * @param {Object} cipherParams An object with any of the possible cipher parameters.\n\t         *\n\t         * @example\n\t         *\n\t         *     var cipherParams = CryptoJS.lib.CipherParams.create({\n\t         *         ciphertext: ciphertextWordArray,\n\t         *         key: keyWordArray,\n\t         *         iv: ivWordArray,\n\t         *         salt: saltWordArray,\n\t         *         algorithm: CryptoJS.algo.AES,\n\t         *         mode: CryptoJS.mode.CBC,\n\t         *         padding: CryptoJS.pad.PKCS7,\n\t         *         blockSize: 4,\n\t         *         formatter: CryptoJS.format.OpenSSL\n\t         *     });\n\t         */\n\t        init: function (cipherParams) {\n\t            this.mixIn(cipherParams);\n\t        },\n\n\t        /**\n\t         * Converts this cipher params object to a string.\n\t         *\n\t         * @param {Format} formatter (Optional) The formatting strategy to use.\n\t         *\n\t         * @return {string} The stringified cipher params.\n\t         *\n\t         * @throws Error If neither the formatter nor the default formatter is set.\n\t         *\n\t         * @example\n\t         *\n\t         *     var string = cipherParams + '';\n\t         *     var string = cipherParams.toString();\n\t         *     var string = cipherParams.toString(CryptoJS.format.OpenSSL);\n\t         */\n\t        toString: function (formatter) {\n\t            return (formatter || this.formatter).stringify(this);\n\t        }\n\t    });\n\n\t    /**\n\t     * Format namespace.\n\t     */\n\t    var C_format = C.format = {};\n\n\t    /**\n\t     * OpenSSL formatting strategy.\n\t     */\n\t    var OpenSSLFormatter = C_format.OpenSSL = {\n\t        /**\n\t         * Converts a cipher params object to an OpenSSL-compatible string.\n\t         *\n\t         * @param {CipherParams} cipherParams The cipher params object.\n\t         *\n\t         * @return {string} The OpenSSL-compatible string.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var openSSLString = CryptoJS.format.OpenSSL.stringify(cipherParams);\n\t         */\n\t        stringify: function (cipherParams) {\n\t            // Shortcuts\n\t            var ciphertext = cipherParams.ciphertext;\n\t            var salt = cipherParams.salt;\n\n\t            // Format\n\t            if (salt) {\n\t                var wordArray = WordArray.create([0x53616c74, 0x65645f5f]).concat(salt).concat(ciphertext);\n\t            } else {\n\t                var wordArray = ciphertext;\n\t            }\n\n\t            return wordArray.toString(Base64);\n\t        },\n\n\t        /**\n\t         * Converts an OpenSSL-compatible string to a cipher params object.\n\t         *\n\t         * @param {string} openSSLStr The OpenSSL-compatible string.\n\t         *\n\t         * @return {CipherParams} The cipher params object.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var cipherParams = CryptoJS.format.OpenSSL.parse(openSSLString);\n\t         */\n\t        parse: function (openSSLStr) {\n\t            // Parse base64\n\t            var ciphertext = Base64.parse(openSSLStr);\n\n\t            // Shortcut\n\t            var ciphertextWords = ciphertext.words;\n\n\t            // Test for salt\n\t            if (ciphertextWords[0] == 0x53616c74 && ciphertextWords[1] == 0x65645f5f) {\n\t                // Extract salt\n\t                var salt = WordArray.create(ciphertextWords.slice(2, 4));\n\n\t                // Remove salt from ciphertext\n\t                ciphertextWords.splice(0, 4);\n\t                ciphertext.sigBytes -= 16;\n\t            }\n\n\t            return CipherParams.create({ ciphertext: ciphertext, salt: salt });\n\t        }\n\t    };\n\n\t    /**\n\t     * A cipher wrapper that returns ciphertext as a serializable cipher params object.\n\t     */\n\t    var SerializableCipher = C_lib.SerializableCipher = Base.extend({\n\t        /**\n\t         * Configuration options.\n\t         *\n\t         * @property {Formatter} format The formatting strategy to convert cipher param objects to and from a string. Default: OpenSSL\n\t         */\n\t        cfg: Base.extend({\n\t            format: OpenSSLFormatter\n\t        }),\n\n\t        /**\n\t         * Encrypts a message.\n\t         *\n\t         * @param {Cipher} cipher The cipher algorithm to use.\n\t         * @param {WordArray|string} message The message to encrypt.\n\t         * @param {WordArray} key The key.\n\t         * @param {Object} cfg (Optional) The configuration options to use for this operation.\n\t         *\n\t         * @return {CipherParams} A cipher params object.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key);\n\t         *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key, { iv: iv });\n\t         *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key, { iv: iv, format: CryptoJS.format.OpenSSL });\n\t         */\n\t        encrypt: function (cipher, message, key, cfg) {\n\t            // Apply config defaults\n\t            cfg = this.cfg.extend(cfg);\n\n\t            // Encrypt\n\t            var encryptor = cipher.createEncryptor(key, cfg);\n\t            var ciphertext = encryptor.finalize(message);\n\n\t            // Shortcut\n\t            var cipherCfg = encryptor.cfg;\n\n\t            // Create and return serializable cipher params\n\t            return CipherParams.create({\n\t                ciphertext: ciphertext,\n\t                key: key,\n\t                iv: cipherCfg.iv,\n\t                algorithm: cipher,\n\t                mode: cipherCfg.mode,\n\t                padding: cipherCfg.padding,\n\t                blockSize: cipher.blockSize,\n\t                formatter: cfg.format\n\t            });\n\t        },\n\n\t        /**\n\t         * Decrypts serialized ciphertext.\n\t         *\n\t         * @param {Cipher} cipher The cipher algorithm to use.\n\t         * @param {CipherParams|string} ciphertext The ciphertext to decrypt.\n\t         * @param {WordArray} key The key.\n\t         * @param {Object} cfg (Optional) The configuration options to use for this operation.\n\t         *\n\t         * @return {WordArray} The plaintext.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var plaintext = CryptoJS.lib.SerializableCipher.decrypt(CryptoJS.algo.AES, formattedCiphertext, key, { iv: iv, format: CryptoJS.format.OpenSSL });\n\t         *     var plaintext = CryptoJS.lib.SerializableCipher.decrypt(CryptoJS.algo.AES, ciphertextParams, key, { iv: iv, format: CryptoJS.format.OpenSSL });\n\t         */\n\t        decrypt: function (cipher, ciphertext, key, cfg) {\n\t            // Apply config defaults\n\t            cfg = this.cfg.extend(cfg);\n\n\t            // Convert string to CipherParams\n\t            ciphertext = this._parse(ciphertext, cfg.format);\n\n\t            // Decrypt\n\t            var plaintext = cipher.createDecryptor(key, cfg).finalize(ciphertext.ciphertext);\n\n\t            return plaintext;\n\t        },\n\n\t        /**\n\t         * Converts serialized ciphertext to CipherParams,\n\t         * else assumed CipherParams already and returns ciphertext unchanged.\n\t         *\n\t         * @param {CipherParams|string} ciphertext The ciphertext.\n\t         * @param {Formatter} format The formatting strategy to use to parse serialized ciphertext.\n\t         *\n\t         * @return {CipherParams} The unserialized ciphertext.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var ciphertextParams = CryptoJS.lib.SerializableCipher._parse(ciphertextStringOrParams, format);\n\t         */\n\t        _parse: function (ciphertext, format) {\n\t            if (typeof ciphertext == 'string') {\n\t                return format.parse(ciphertext, this);\n\t            } else {\n\t                return ciphertext;\n\t            }\n\t        }\n\t    });\n\n\t    /**\n\t     * Key derivation function namespace.\n\t     */\n\t    var C_kdf = C.kdf = {};\n\n\t    /**\n\t     * OpenSSL key derivation function.\n\t     */\n\t    var OpenSSLKdf = C_kdf.OpenSSL = {\n\t        /**\n\t         * Derives a key and IV from a password.\n\t         *\n\t         * @param {string} password The password to derive from.\n\t         * @param {number} keySize The size in words of the key to generate.\n\t         * @param {number} ivSize The size in words of the IV to generate.\n\t         * @param {WordArray|string} salt (Optional) A 64-bit salt to use. If omitted, a salt will be generated randomly.\n\t         *\n\t         * @return {CipherParams} A cipher params object with the key, IV, and salt.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var derivedParams = CryptoJS.kdf.OpenSSL.execute('Password', 256/32, 128/32);\n\t         *     var derivedParams = CryptoJS.kdf.OpenSSL.execute('Password', 256/32, 128/32, 'saltsalt');\n\t         */\n\t        execute: function (password, keySize, ivSize, salt) {\n\t            // Generate random salt\n\t            if (!salt) {\n\t                salt = WordArray.random(64/8);\n\t            }\n\n\t            // Derive key and IV\n\t            var key = EvpKDF.create({ keySize: keySize + ivSize }).compute(password, salt);\n\n\t            // Separate key and IV\n\t            var iv = WordArray.create(key.words.slice(keySize), ivSize * 4);\n\t            key.sigBytes = keySize * 4;\n\n\t            // Return params\n\t            return CipherParams.create({ key: key, iv: iv, salt: salt });\n\t        }\n\t    };\n\n\t    /**\n\t     * A serializable cipher wrapper that derives the key from a password,\n\t     * and returns ciphertext as a serializable cipher params object.\n\t     */\n\t    var PasswordBasedCipher = C_lib.PasswordBasedCipher = SerializableCipher.extend({\n\t        /**\n\t         * Configuration options.\n\t         *\n\t         * @property {KDF} kdf The key derivation function to use to generate a key and IV from a password. Default: OpenSSL\n\t         */\n\t        cfg: SerializableCipher.cfg.extend({\n\t            kdf: OpenSSLKdf\n\t        }),\n\n\t        /**\n\t         * Encrypts a message using a password.\n\t         *\n\t         * @param {Cipher} cipher The cipher algorithm to use.\n\t         * @param {WordArray|string} message The message to encrypt.\n\t         * @param {string} password The password.\n\t         * @param {Object} cfg (Optional) The configuration options to use for this operation.\n\t         *\n\t         * @return {CipherParams} A cipher params object.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var ciphertextParams = CryptoJS.lib.PasswordBasedCipher.encrypt(CryptoJS.algo.AES, message, 'password');\n\t         *     var ciphertextParams = CryptoJS.lib.PasswordBasedCipher.encrypt(CryptoJS.algo.AES, message, 'password', { format: CryptoJS.format.OpenSSL });\n\t         */\n\t        encrypt: function (cipher, message, password, cfg) {\n\t            // Apply config defaults\n\t            cfg = this.cfg.extend(cfg);\n\n\t            // Derive key and other params\n\t            var derivedParams = cfg.kdf.execute(password, cipher.keySize, cipher.ivSize);\n\n\t            // Add IV to config\n\t            cfg.iv = derivedParams.iv;\n\n\t            // Encrypt\n\t            var ciphertext = SerializableCipher.encrypt.call(this, cipher, message, derivedParams.key, cfg);\n\n\t            // Mix in derived params\n\t            ciphertext.mixIn(derivedParams);\n\n\t            return ciphertext;\n\t        },\n\n\t        /**\n\t         * Decrypts serialized ciphertext using a password.\n\t         *\n\t         * @param {Cipher} cipher The cipher algorithm to use.\n\t         * @param {CipherParams|string} ciphertext The ciphertext to decrypt.\n\t         * @param {string} password The password.\n\t         * @param {Object} cfg (Optional) The configuration options to use for this operation.\n\t         *\n\t         * @return {WordArray} The plaintext.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var plaintext = CryptoJS.lib.PasswordBasedCipher.decrypt(CryptoJS.algo.AES, formattedCiphertext, 'password', { format: CryptoJS.format.OpenSSL });\n\t         *     var plaintext = CryptoJS.lib.PasswordBasedCipher.decrypt(CryptoJS.algo.AES, ciphertextParams, 'password', { format: CryptoJS.format.OpenSSL });\n\t         */\n\t        decrypt: function (cipher, ciphertext, password, cfg) {\n\t            // Apply config defaults\n\t            cfg = this.cfg.extend(cfg);\n\n\t            // Convert string to CipherParams\n\t            ciphertext = this._parse(ciphertext, cfg.format);\n\n\t            // Derive key and other params\n\t            var derivedParams = cfg.kdf.execute(password, cipher.keySize, cipher.ivSize, ciphertext.salt);\n\n\t            // Add IV to config\n\t            cfg.iv = derivedParams.iv;\n\n\t            // Decrypt\n\t            var plaintext = SerializableCipher.decrypt.call(this, cipher, ciphertext, derivedParams.key, cfg);\n\n\t            return plaintext;\n\t        }\n\t    });\n\t}());\n\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory();\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\troot.CryptoJS = factory();\n\t}\n}(this, function () {\n\n\t/**\n\t * CryptoJS core components.\n\t */\n\tvar CryptoJS = CryptoJS || (function (Math, undefined) {\n\t    /*\n\t     * Local polyfil of Object.create\n\t     */\n\t    var create = Object.create || (function () {\n\t        function F() {};\n\n\t        return function (obj) {\n\t            var subtype;\n\n\t            F.prototype = obj;\n\n\t            subtype = new F();\n\n\t            F.prototype = null;\n\n\t            return subtype;\n\t        };\n\t    }())\n\n\t    /**\n\t     * CryptoJS namespace.\n\t     */\n\t    var C = {};\n\n\t    /**\n\t     * Library namespace.\n\t     */\n\t    var C_lib = C.lib = {};\n\n\t    /**\n\t     * Base object for prototypal inheritance.\n\t     */\n\t    var Base = C_lib.Base = (function () {\n\n\n\t        return {\n\t            /**\n\t             * Creates a new object that inherits from this object.\n\t             *\n\t             * @param {Object} overrides Properties to copy into the new object.\n\t             *\n\t             * @return {Object} The new object.\n\t             *\n\t             * @static\n\t             *\n\t             * @example\n\t             *\n\t             *     var MyType = CryptoJS.lib.Base.extend({\n\t             *         field: 'value',\n\t             *\n\t             *         method: function () {\n\t             *         }\n\t             *     });\n\t             */\n\t            extend: function (overrides) {\n\t                // Spawn\n\t                var subtype = create(this);\n\n\t                // Augment\n\t                if (overrides) {\n\t                    subtype.mixIn(overrides);\n\t                }\n\n\t                // Create default initializer\n\t                if (!subtype.hasOwnProperty('init') || this.init === subtype.init) {\n\t                    subtype.init = function () {\n\t                        subtype.$super.init.apply(this, arguments);\n\t                    };\n\t                }\n\n\t                // Initializer's prototype is the subtype object\n\t                subtype.init.prototype = subtype;\n\n\t                // Reference supertype\n\t                subtype.$super = this;\n\n\t                return subtype;\n\t            },\n\n\t            /**\n\t             * Extends this object and runs the init method.\n\t             * Arguments to create() will be passed to init().\n\t             *\n\t             * @return {Object} The new object.\n\t             *\n\t             * @static\n\t             *\n\t             * @example\n\t             *\n\t             *     var instance = MyType.create();\n\t             */\n\t            create: function () {\n\t                var instance = this.extend();\n\t                instance.init.apply(instance, arguments);\n\n\t                return instance;\n\t            },\n\n\t            /**\n\t             * Initializes a newly created object.\n\t             * Override this method to add some logic when your objects are created.\n\t             *\n\t             * @example\n\t             *\n\t             *     var MyType = CryptoJS.lib.Base.extend({\n\t             *         init: function () {\n\t             *             // ...\n\t             *         }\n\t             *     });\n\t             */\n\t            init: function () {\n\t            },\n\n\t            /**\n\t             * Copies properties into this object.\n\t             *\n\t             * @param {Object} properties The properties to mix in.\n\t             *\n\t             * @example\n\t             *\n\t             *     MyType.mixIn({\n\t             *         field: 'value'\n\t             *     });\n\t             */\n\t            mixIn: function (properties) {\n\t                for (var propertyName in properties) {\n\t                    if (properties.hasOwnProperty(propertyName)) {\n\t                        this[propertyName] = properties[propertyName];\n\t                    }\n\t                }\n\n\t                // IE won't copy toString using the loop above\n\t                if (properties.hasOwnProperty('toString')) {\n\t                    this.toString = properties.toString;\n\t                }\n\t            },\n\n\t            /**\n\t             * Creates a copy of this object.\n\t             *\n\t             * @return {Object} The clone.\n\t             *\n\t             * @example\n\t             *\n\t             *     var clone = instance.clone();\n\t             */\n\t            clone: function () {\n\t                return this.init.prototype.extend(this);\n\t            }\n\t        };\n\t    }());\n\n\t    /**\n\t     * An array of 32-bit words.\n\t     *\n\t     * @property {Array} words The array of 32-bit words.\n\t     * @property {number} sigBytes The number of significant bytes in this word array.\n\t     */\n\t    var WordArray = C_lib.WordArray = Base.extend({\n\t        /**\n\t         * Initializes a newly created word array.\n\t         *\n\t         * @param {Array} words (Optional) An array of 32-bit words.\n\t         * @param {number} sigBytes (Optional) The number of significant bytes in the words.\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.lib.WordArray.create();\n\t         *     var wordArray = CryptoJS.lib.WordArray.create([0x00010203, 0x04050607]);\n\t         *     var wordArray = CryptoJS.lib.WordArray.create([0x00010203, 0x04050607], 6);\n\t         */\n\t        init: function (words, sigBytes) {\n\t            words = this.words = words || [];\n\n\t            if (sigBytes != undefined) {\n\t                this.sigBytes = sigBytes;\n\t            } else {\n\t                this.sigBytes = words.length * 4;\n\t            }\n\t        },\n\n\t        /**\n\t         * Converts this word array to a string.\n\t         *\n\t         * @param {Encoder} encoder (Optional) The encoding strategy to use. Default: CryptoJS.enc.Hex\n\t         *\n\t         * @return {string} The stringified word array.\n\t         *\n\t         * @example\n\t         *\n\t         *     var string = wordArray + '';\n\t         *     var string = wordArray.toString();\n\t         *     var string = wordArray.toString(CryptoJS.enc.Utf8);\n\t         */\n\t        toString: function (encoder) {\n\t            return (encoder || Hex).stringify(this);\n\t        },\n\n\t        /**\n\t         * Concatenates a word array to this word array.\n\t         *\n\t         * @param {WordArray} wordArray The word array to append.\n\t         *\n\t         * @return {WordArray} This word array.\n\t         *\n\t         * @example\n\t         *\n\t         *     wordArray1.concat(wordArray2);\n\t         */\n\t        concat: function (wordArray) {\n\t            // Shortcuts\n\t            var thisWords = this.words;\n\t            var thatWords = wordArray.words;\n\t            var thisSigBytes = this.sigBytes;\n\t            var thatSigBytes = wordArray.sigBytes;\n\n\t            // Clamp excess bits\n\t            this.clamp();\n\n\t            // Concat\n\t            if (thisSigBytes % 4) {\n\t                // Copy one byte at a time\n\t                for (var i = 0; i < thatSigBytes; i++) {\n\t                    var thatByte = (thatWords[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;\n\t                    thisWords[(thisSigBytes + i) >>> 2] |= thatByte << (24 - ((thisSigBytes + i) % 4) * 8);\n\t                }\n\t            } else {\n\t                // Copy one word at a time\n\t                for (var i = 0; i < thatSigBytes; i += 4) {\n\t                    thisWords[(thisSigBytes + i) >>> 2] = thatWords[i >>> 2];\n\t                }\n\t            }\n\t            this.sigBytes += thatSigBytes;\n\n\t            // Chainable\n\t            return this;\n\t        },\n\n\t        /**\n\t         * Removes insignificant bits.\n\t         *\n\t         * @example\n\t         *\n\t         *     wordArray.clamp();\n\t         */\n\t        clamp: function () {\n\t            // Shortcuts\n\t            var words = this.words;\n\t            var sigBytes = this.sigBytes;\n\n\t            // Clamp\n\t            words[sigBytes >>> 2] &= 0xffffffff << (32 - (sigBytes % 4) * 8);\n\t            words.length = Math.ceil(sigBytes / 4);\n\t        },\n\n\t        /**\n\t         * Creates a copy of this word array.\n\t         *\n\t         * @return {WordArray} The clone.\n\t         *\n\t         * @example\n\t         *\n\t         *     var clone = wordArray.clone();\n\t         */\n\t        clone: function () {\n\t            var clone = Base.clone.call(this);\n\t            clone.words = this.words.slice(0);\n\n\t            return clone;\n\t        },\n\n\t        /**\n\t         * Creates a word array filled with random bytes.\n\t         *\n\t         * @param {number} nBytes The number of random bytes to generate.\n\t         *\n\t         * @return {WordArray} The random word array.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.lib.WordArray.random(16);\n\t         */\n\t        random: function (nBytes) {\n\t            var words = [];\n\n\t            var r = (function (m_w) {\n\t                var m_w = m_w;\n\t                var m_z = 0x3ade68b1;\n\t                var mask = 0xffffffff;\n\n\t                return function () {\n\t                    m_z = (0x9069 * (m_z & 0xFFFF) + (m_z >> 0x10)) & mask;\n\t                    m_w = (0x4650 * (m_w & 0xFFFF) + (m_w >> 0x10)) & mask;\n\t                    var result = ((m_z << 0x10) + m_w) & mask;\n\t                    result /= 0x100000000;\n\t                    result += 0.5;\n\t                    return result * (Math.random() > .5 ? 1 : -1);\n\t                }\n\t            });\n\n\t            for (var i = 0, rcache; i < nBytes; i += 4) {\n\t                var _r = r((rcache || Math.random()) * 0x100000000);\n\n\t                rcache = _r() * 0x3ade67b7;\n\t                words.push((_r() * 0x100000000) | 0);\n\t            }\n\n\t            return new WordArray.init(words, nBytes);\n\t        }\n\t    });\n\n\t    /**\n\t     * Encoder namespace.\n\t     */\n\t    var C_enc = C.enc = {};\n\n\t    /**\n\t     * Hex encoding strategy.\n\t     */\n\t    var Hex = C_enc.Hex = {\n\t        /**\n\t         * Converts a word array to a hex string.\n\t         *\n\t         * @param {WordArray} wordArray The word array.\n\t         *\n\t         * @return {string} The hex string.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var hexString = CryptoJS.enc.Hex.stringify(wordArray);\n\t         */\n\t        stringify: function (wordArray) {\n\t            // Shortcuts\n\t            var words = wordArray.words;\n\t            var sigBytes = wordArray.sigBytes;\n\n\t            // Convert\n\t            var hexChars = [];\n\t            for (var i = 0; i < sigBytes; i++) {\n\t                var bite = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;\n\t                hexChars.push((bite >>> 4).toString(16));\n\t                hexChars.push((bite & 0x0f).toString(16));\n\t            }\n\n\t            return hexChars.join('');\n\t        },\n\n\t        /**\n\t         * Converts a hex string to a word array.\n\t         *\n\t         * @param {string} hexStr The hex string.\n\t         *\n\t         * @return {WordArray} The word array.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.enc.Hex.parse(hexString);\n\t         */\n\t        parse: function (hexStr) {\n\t            // Shortcut\n\t            var hexStrLength = hexStr.length;\n\n\t            // Convert\n\t            var words = [];\n\t            for (var i = 0; i < hexStrLength; i += 2) {\n\t                words[i >>> 3] |= parseInt(hexStr.substr(i, 2), 16) << (24 - (i % 8) * 4);\n\t            }\n\n\t            return new WordArray.init(words, hexStrLength / 2);\n\t        }\n\t    };\n\n\t    /**\n\t     * Latin1 encoding strategy.\n\t     */\n\t    var Latin1 = C_enc.Latin1 = {\n\t        /**\n\t         * Converts a word array to a Latin1 string.\n\t         *\n\t         * @param {WordArray} wordArray The word array.\n\t         *\n\t         * @return {string} The Latin1 string.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var latin1String = CryptoJS.enc.Latin1.stringify(wordArray);\n\t         */\n\t        stringify: function (wordArray) {\n\t            // Shortcuts\n\t            var words = wordArray.words;\n\t            var sigBytes = wordArray.sigBytes;\n\n\t            // Convert\n\t            var latin1Chars = [];\n\t            for (var i = 0; i < sigBytes; i++) {\n\t                var bite = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;\n\t                latin1Chars.push(String.fromCharCode(bite));\n\t            }\n\n\t            return latin1Chars.join('');\n\t        },\n\n\t        /**\n\t         * Converts a Latin1 string to a word array.\n\t         *\n\t         * @param {string} latin1Str The Latin1 string.\n\t         *\n\t         * @return {WordArray} The word array.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.enc.Latin1.parse(latin1String);\n\t         */\n\t        parse: function (latin1Str) {\n\t            // Shortcut\n\t            var latin1StrLength = latin1Str.length;\n\n\t            // Convert\n\t            var words = [];\n\t            for (var i = 0; i < latin1StrLength; i++) {\n\t                words[i >>> 2] |= (latin1Str.charCodeAt(i) & 0xff) << (24 - (i % 4) * 8);\n\t            }\n\n\t            return new WordArray.init(words, latin1StrLength);\n\t        }\n\t    };\n\n\t    /**\n\t     * UTF-8 encoding strategy.\n\t     */\n\t    var Utf8 = C_enc.Utf8 = {\n\t        /**\n\t         * Converts a word array to a UTF-8 string.\n\t         *\n\t         * @param {WordArray} wordArray The word array.\n\t         *\n\t         * @return {string} The UTF-8 string.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var utf8String = CryptoJS.enc.Utf8.stringify(wordArray);\n\t         */\n\t        stringify: function (wordArray) {\n\t            try {\n\t                return decodeURIComponent(escape(Latin1.stringify(wordArray)));\n\t            } catch (e) {\n\t                throw new Error('Malformed UTF-8 data');\n\t            }\n\t        },\n\n\t        /**\n\t         * Converts a UTF-8 string to a word array.\n\t         *\n\t         * @param {string} utf8Str The UTF-8 string.\n\t         *\n\t         * @return {WordArray} The word array.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.enc.Utf8.parse(utf8String);\n\t         */\n\t        parse: function (utf8Str) {\n\t            return Latin1.parse(unescape(encodeURIComponent(utf8Str)));\n\t        }\n\t    };\n\n\t    /**\n\t     * Abstract buffered block algorithm template.\n\t     *\n\t     * The property blockSize must be implemented in a concrete subtype.\n\t     *\n\t     * @property {number} _minBufferSize The number of blocks that should be kept unprocessed in the buffer. Default: 0\n\t     */\n\t    var BufferedBlockAlgorithm = C_lib.BufferedBlockAlgorithm = Base.extend({\n\t        /**\n\t         * Resets this block algorithm's data buffer to its initial state.\n\t         *\n\t         * @example\n\t         *\n\t         *     bufferedBlockAlgorithm.reset();\n\t         */\n\t        reset: function () {\n\t            // Initial values\n\t            this._data = new WordArray.init();\n\t            this._nDataBytes = 0;\n\t        },\n\n\t        /**\n\t         * Adds new data to this block algorithm's buffer.\n\t         *\n\t         * @param {WordArray|string} data The data to append. Strings are converted to a WordArray using UTF-8.\n\t         *\n\t         * @example\n\t         *\n\t         *     bufferedBlockAlgorithm._append('data');\n\t         *     bufferedBlockAlgorithm._append(wordArray);\n\t         */\n\t        _append: function (data) {\n\t            // Convert string to WordArray, else assume WordArray already\n\t            if (typeof data == 'string') {\n\t                data = Utf8.parse(data);\n\t            }\n\n\t            // Append\n\t            this._data.concat(data);\n\t            this._nDataBytes += data.sigBytes;\n\t        },\n\n\t        /**\n\t         * Processes available data blocks.\n\t         *\n\t         * This method invokes _doProcessBlock(offset), which must be implemented by a concrete subtype.\n\t         *\n\t         * @param {boolean} doFlush Whether all blocks and partial blocks should be processed.\n\t         *\n\t         * @return {WordArray} The processed data.\n\t         *\n\t         * @example\n\t         *\n\t         *     var processedData = bufferedBlockAlgorithm._process();\n\t         *     var processedData = bufferedBlockAlgorithm._process(!!'flush');\n\t         */\n\t        _process: function (doFlush) {\n\t            // Shortcuts\n\t            var data = this._data;\n\t            var dataWords = data.words;\n\t            var dataSigBytes = data.sigBytes;\n\t            var blockSize = this.blockSize;\n\t            var blockSizeBytes = blockSize * 4;\n\n\t            // Count blocks ready\n\t            var nBlocksReady = dataSigBytes / blockSizeBytes;\n\t            if (doFlush) {\n\t                // Round up to include partial blocks\n\t                nBlocksReady = Math.ceil(nBlocksReady);\n\t            } else {\n\t                // Round down to include only full blocks,\n\t                // less the number of blocks that must remain in the buffer\n\t                nBlocksReady = Math.max((nBlocksReady | 0) - this._minBufferSize, 0);\n\t            }\n\n\t            // Count words ready\n\t            var nWordsReady = nBlocksReady * blockSize;\n\n\t            // Count bytes ready\n\t            var nBytesReady = Math.min(nWordsReady * 4, dataSigBytes);\n\n\t            // Process blocks\n\t            if (nWordsReady) {\n\t                for (var offset = 0; offset < nWordsReady; offset += blockSize) {\n\t                    // Perform concrete-algorithm logic\n\t                    this._doProcessBlock(dataWords, offset);\n\t                }\n\n\t                // Remove processed words\n\t                var processedWords = dataWords.splice(0, nWordsReady);\n\t                data.sigBytes -= nBytesReady;\n\t            }\n\n\t            // Return processed words\n\t            return new WordArray.init(processedWords, nBytesReady);\n\t        },\n\n\t        /**\n\t         * Creates a copy of this object.\n\t         *\n\t         * @return {Object} The clone.\n\t         *\n\t         * @example\n\t         *\n\t         *     var clone = bufferedBlockAlgorithm.clone();\n\t         */\n\t        clone: function () {\n\t            var clone = Base.clone.call(this);\n\t            clone._data = this._data.clone();\n\n\t            return clone;\n\t        },\n\n\t        _minBufferSize: 0\n\t    });\n\n\t    /**\n\t     * Abstract hasher template.\n\t     *\n\t     * @property {number} blockSize The number of 32-bit words this hasher operates on. Default: 16 (512 bits)\n\t     */\n\t    var Hasher = C_lib.Hasher = BufferedBlockAlgorithm.extend({\n\t        /**\n\t         * Configuration options.\n\t         */\n\t        cfg: Base.extend(),\n\n\t        /**\n\t         * Initializes a newly created hasher.\n\t         *\n\t         * @param {Object} cfg (Optional) The configuration options to use for this hash computation.\n\t         *\n\t         * @example\n\t         *\n\t         *     var hasher = CryptoJS.algo.SHA256.create();\n\t         */\n\t        init: function (cfg) {\n\t            // Apply config defaults\n\t            this.cfg = this.cfg.extend(cfg);\n\n\t            // Set initial values\n\t            this.reset();\n\t        },\n\n\t        /**\n\t         * Resets this hasher to its initial state.\n\t         *\n\t         * @example\n\t         *\n\t         *     hasher.reset();\n\t         */\n\t        reset: function () {\n\t            // Reset data buffer\n\t            BufferedBlockAlgorithm.reset.call(this);\n\n\t            // Perform concrete-hasher logic\n\t            this._doReset();\n\t        },\n\n\t        /**\n\t         * Updates this hasher with a message.\n\t         *\n\t         * @param {WordArray|string} messageUpdate The message to append.\n\t         *\n\t         * @return {Hasher} This hasher.\n\t         *\n\t         * @example\n\t         *\n\t         *     hasher.update('message');\n\t         *     hasher.update(wordArray);\n\t         */\n\t        update: function (messageUpdate) {\n\t            // Append\n\t            this._append(messageUpdate);\n\n\t            // Update the hash\n\t            this._process();\n\n\t            // Chainable\n\t            return this;\n\t        },\n\n\t        /**\n\t         * Finalizes the hash computation.\n\t         * Note that the finalize operation is effectively a destructive, read-once operation.\n\t         *\n\t         * @param {WordArray|string} messageUpdate (Optional) A final message update.\n\t         *\n\t         * @return {WordArray} The hash.\n\t         *\n\t         * @example\n\t         *\n\t         *     var hash = hasher.finalize();\n\t         *     var hash = hasher.finalize('message');\n\t         *     var hash = hasher.finalize(wordArray);\n\t         */\n\t        finalize: function (messageUpdate) {\n\t            // Final message update\n\t            if (messageUpdate) {\n\t                this._append(messageUpdate);\n\t            }\n\n\t            // Perform concrete-hasher logic\n\t            var hash = this._doFinalize();\n\n\t            return hash;\n\t        },\n\n\t        blockSize: 512/32,\n\n\t        /**\n\t         * Creates a shortcut function to a hasher's object interface.\n\t         *\n\t         * @param {Hasher} hasher The hasher to create a helper for.\n\t         *\n\t         * @return {Function} The shortcut function.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var SHA256 = CryptoJS.lib.Hasher._createHelper(CryptoJS.algo.SHA256);\n\t         */\n\t        _createHelper: function (hasher) {\n\t            return function (message, cfg) {\n\t                return new hasher.init(cfg).finalize(message);\n\t            };\n\t        },\n\n\t        /**\n\t         * Creates a shortcut function to the HMAC's object interface.\n\t         *\n\t         * @param {Hasher} hasher The hasher to use in this HMAC helper.\n\t         *\n\t         * @return {Function} The shortcut function.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var HmacSHA256 = CryptoJS.lib.Hasher._createHmacHelper(CryptoJS.algo.SHA256);\n\t         */\n\t        _createHmacHelper: function (hasher) {\n\t            return function (message, key) {\n\t                return new C_algo.HMAC.init(hasher, key).finalize(message);\n\t            };\n\t        }\n\t    });\n\n\t    /**\n\t     * Algorithm namespace.\n\t     */\n\t    var C_algo = C.algo = {};\n\n\t    return C;\n\t}(Math));\n\n\n\treturn CryptoJS;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var C_enc = C.enc;\n\n\t    /**\n\t     * Base64 encoding strategy.\n\t     */\n\t    var Base64 = C_enc.Base64 = {\n\t        /**\n\t         * Converts a word array to a Base64 string.\n\t         *\n\t         * @param {WordArray} wordArray The word array.\n\t         *\n\t         * @return {string} The Base64 string.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var base64String = CryptoJS.enc.Base64.stringify(wordArray);\n\t         */\n\t        stringify: function (wordArray) {\n\t            // Shortcuts\n\t            var words = wordArray.words;\n\t            var sigBytes = wordArray.sigBytes;\n\t            var map = this._map;\n\n\t            // Clamp excess bits\n\t            wordArray.clamp();\n\n\t            // Convert\n\t            var base64Chars = [];\n\t            for (var i = 0; i < sigBytes; i += 3) {\n\t                var byte1 = (words[i >>> 2]       >>> (24 - (i % 4) * 8))       & 0xff;\n\t                var byte2 = (words[(i + 1) >>> 2] >>> (24 - ((i + 1) % 4) * 8)) & 0xff;\n\t                var byte3 = (words[(i + 2) >>> 2] >>> (24 - ((i + 2) % 4) * 8)) & 0xff;\n\n\t                var triplet = (byte1 << 16) | (byte2 << 8) | byte3;\n\n\t                for (var j = 0; (j < 4) && (i + j * 0.75 < sigBytes); j++) {\n\t                    base64Chars.push(map.charAt((triplet >>> (6 * (3 - j))) & 0x3f));\n\t                }\n\t            }\n\n\t            // Add padding\n\t            var paddingChar = map.charAt(64);\n\t            if (paddingChar) {\n\t                while (base64Chars.length % 4) {\n\t                    base64Chars.push(paddingChar);\n\t                }\n\t            }\n\n\t            return base64Chars.join('');\n\t        },\n\n\t        /**\n\t         * Converts a Base64 string to a word array.\n\t         *\n\t         * @param {string} base64Str The Base64 string.\n\t         *\n\t         * @return {WordArray} The word array.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.enc.Base64.parse(base64String);\n\t         */\n\t        parse: function (base64Str) {\n\t            // Shortcuts\n\t            var base64StrLength = base64Str.length;\n\t            var map = this._map;\n\t            var reverseMap = this._reverseMap;\n\n\t            if (!reverseMap) {\n\t                    reverseMap = this._reverseMap = [];\n\t                    for (var j = 0; j < map.length; j++) {\n\t                        reverseMap[map.charCodeAt(j)] = j;\n\t                    }\n\t            }\n\n\t            // Ignore padding\n\t            var paddingChar = map.charAt(64);\n\t            if (paddingChar) {\n\t                var paddingIndex = base64Str.indexOf(paddingChar);\n\t                if (paddingIndex !== -1) {\n\t                    base64StrLength = paddingIndex;\n\t                }\n\t            }\n\n\t            // Convert\n\t            return parseLoop(base64Str, base64StrLength, reverseMap);\n\n\t        },\n\n\t        _map: 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/='\n\t    };\n\n\t    function parseLoop(base64Str, base64StrLength, reverseMap) {\n\t      var words = [];\n\t      var nBytes = 0;\n\t      for (var i = 0; i < base64StrLength; i++) {\n\t          if (i % 4) {\n\t              var bits1 = reverseMap[base64Str.charCodeAt(i - 1)] << ((i % 4) * 2);\n\t              var bits2 = reverseMap[base64Str.charCodeAt(i)] >>> (6 - (i % 4) * 2);\n\t              words[nBytes >>> 2] |= (bits1 | bits2) << (24 - (nBytes % 4) * 8);\n\t              nBytes++;\n\t          }\n\t      }\n\t      return WordArray.create(words, nBytes);\n\t    }\n\t}());\n\n\n\treturn CryptoJS.enc.Base64;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var C_enc = C.enc;\n\n\t    /**\n\t     * UTF-16 BE encoding strategy.\n\t     */\n\t    var Utf16BE = C_enc.Utf16 = C_enc.Utf16BE = {\n\t        /**\n\t         * Converts a word array to a UTF-16 BE string.\n\t         *\n\t         * @param {WordArray} wordArray The word array.\n\t         *\n\t         * @return {string} The UTF-16 BE string.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var utf16String = CryptoJS.enc.Utf16.stringify(wordArray);\n\t         */\n\t        stringify: function (wordArray) {\n\t            // Shortcuts\n\t            var words = wordArray.words;\n\t            var sigBytes = wordArray.sigBytes;\n\n\t            // Convert\n\t            var utf16Chars = [];\n\t            for (var i = 0; i < sigBytes; i += 2) {\n\t                var codePoint = (words[i >>> 2] >>> (16 - (i % 4) * 8)) & 0xffff;\n\t                utf16Chars.push(String.fromCharCode(codePoint));\n\t            }\n\n\t            return utf16Chars.join('');\n\t        },\n\n\t        /**\n\t         * Converts a UTF-16 BE string to a word array.\n\t         *\n\t         * @param {string} utf16Str The UTF-16 BE string.\n\t         *\n\t         * @return {WordArray} The word array.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.enc.Utf16.parse(utf16String);\n\t         */\n\t        parse: function (utf16Str) {\n\t            // Shortcut\n\t            var utf16StrLength = utf16Str.length;\n\n\t            // Convert\n\t            var words = [];\n\t            for (var i = 0; i < utf16StrLength; i++) {\n\t                words[i >>> 1] |= utf16Str.charCodeAt(i) << (16 - (i % 2) * 16);\n\t            }\n\n\t            return WordArray.create(words, utf16StrLength * 2);\n\t        }\n\t    };\n\n\t    /**\n\t     * UTF-16 LE encoding strategy.\n\t     */\n\t    C_enc.Utf16LE = {\n\t        /**\n\t         * Converts a word array to a UTF-16 LE string.\n\t         *\n\t         * @param {WordArray} wordArray The word array.\n\t         *\n\t         * @return {string} The UTF-16 LE string.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var utf16Str = CryptoJS.enc.Utf16LE.stringify(wordArray);\n\t         */\n\t        stringify: function (wordArray) {\n\t            // Shortcuts\n\t            var words = wordArray.words;\n\t            var sigBytes = wordArray.sigBytes;\n\n\t            // Convert\n\t            var utf16Chars = [];\n\t            for (var i = 0; i < sigBytes; i += 2) {\n\t                var codePoint = swapEndian((words[i >>> 2] >>> (16 - (i % 4) * 8)) & 0xffff);\n\t                utf16Chars.push(String.fromCharCode(codePoint));\n\t            }\n\n\t            return utf16Chars.join('');\n\t        },\n\n\t        /**\n\t         * Converts a UTF-16 LE string to a word array.\n\t         *\n\t         * @param {string} utf16Str The UTF-16 LE string.\n\t         *\n\t         * @return {WordArray} The word array.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.enc.Utf16LE.parse(utf16Str);\n\t         */\n\t        parse: function (utf16Str) {\n\t            // Shortcut\n\t            var utf16StrLength = utf16Str.length;\n\n\t            // Convert\n\t            var words = [];\n\t            for (var i = 0; i < utf16StrLength; i++) {\n\t                words[i >>> 1] |= swapEndian(utf16Str.charCodeAt(i) << (16 - (i % 2) * 16));\n\t            }\n\n\t            return WordArray.create(words, utf16StrLength * 2);\n\t        }\n\t    };\n\n\t    function swapEndian(word) {\n\t        return ((word << 8) & 0xff00ff00) | ((word >>> 8) & 0x00ff00ff);\n\t    }\n\t}());\n\n\n\treturn CryptoJS.enc.Utf16;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./sha1\"), require(\"./hmac\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./sha1\", \"./hmac\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var Base = C_lib.Base;\n\t    var WordArray = C_lib.WordArray;\n\t    var C_algo = C.algo;\n\t    var MD5 = C_algo.MD5;\n\n\t    /**\n\t     * This key derivation function is meant to conform with EVP_BytesToKey.\n\t     * www.openssl.org/docs/crypto/EVP_BytesToKey.html\n\t     */\n\t    var EvpKDF = C_algo.EvpKDF = Base.extend({\n\t        /**\n\t         * Configuration options.\n\t         *\n\t         * @property {number} keySize The key size in words to generate. Default: 4 (128 bits)\n\t         * @property {Hasher} hasher The hash algorithm to use. Default: MD5\n\t         * @property {number} iterations The number of iterations to perform. Default: 1\n\t         */\n\t        cfg: Base.extend({\n\t            keySize: 128/32,\n\t            hasher: MD5,\n\t            iterations: 1\n\t        }),\n\n\t        /**\n\t         * Initializes a newly created key derivation function.\n\t         *\n\t         * @param {Object} cfg (Optional) The configuration options to use for the derivation.\n\t         *\n\t         * @example\n\t         *\n\t         *     var kdf = CryptoJS.algo.EvpKDF.create();\n\t         *     var kdf = CryptoJS.algo.EvpKDF.create({ keySize: 8 });\n\t         *     var kdf = CryptoJS.algo.EvpKDF.create({ keySize: 8, iterations: 1000 });\n\t         */\n\t        init: function (cfg) {\n\t            this.cfg = this.cfg.extend(cfg);\n\t        },\n\n\t        /**\n\t         * Derives a key from a password.\n\t         *\n\t         * @param {WordArray|string} password The password.\n\t         * @param {WordArray|string} salt A salt.\n\t         *\n\t         * @return {WordArray} The derived key.\n\t         *\n\t         * @example\n\t         *\n\t         *     var key = kdf.compute(password, salt);\n\t         */\n\t        compute: function (password, salt) {\n\t            // Shortcut\n\t            var cfg = this.cfg;\n\n\t            // Init hasher\n\t            var hasher = cfg.hasher.create();\n\n\t            // Initial values\n\t            var derivedKey = WordArray.create();\n\n\t            // Shortcuts\n\t            var derivedKeyWords = derivedKey.words;\n\t            var keySize = cfg.keySize;\n\t            var iterations = cfg.iterations;\n\n\t            // Generate key\n\t            while (derivedKeyWords.length < keySize) {\n\t                if (block) {\n\t                    hasher.update(block);\n\t                }\n\t                var block = hasher.update(password).finalize(salt);\n\t                hasher.reset();\n\n\t                // Iterations\n\t                for (var i = 1; i < iterations; i++) {\n\t                    block = hasher.finalize(block);\n\t                    hasher.reset();\n\t                }\n\n\t                derivedKey.concat(block);\n\t            }\n\t            derivedKey.sigBytes = keySize * 4;\n\n\t            return derivedKey;\n\t        }\n\t    });\n\n\t    /**\n\t     * Derives a key from a password.\n\t     *\n\t     * @param {WordArray|string} password The password.\n\t     * @param {WordArray|string} salt A salt.\n\t     * @param {Object} cfg (Optional) The configuration options to use for this computation.\n\t     *\n\t     * @return {WordArray} The derived key.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var key = CryptoJS.EvpKDF(password, salt);\n\t     *     var key = CryptoJS.EvpKDF(password, salt, { keySize: 8 });\n\t     *     var key = CryptoJS.EvpKDF(password, salt, { keySize: 8, iterations: 1000 });\n\t     */\n\t    C.EvpKDF = function (password, salt, cfg) {\n\t        return EvpKDF.create(cfg).compute(password, salt);\n\t    };\n\t}());\n\n\n\treturn CryptoJS.EvpKDF;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function (undefined) {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var CipherParams = C_lib.CipherParams;\n\t    var C_enc = C.enc;\n\t    var Hex = C_enc.Hex;\n\t    var C_format = C.format;\n\n\t    var HexFormatter = C_format.Hex = {\n\t        /**\n\t         * Converts the ciphertext of a cipher params object to a hexadecimally encoded string.\n\t         *\n\t         * @param {CipherParams} cipherParams The cipher params object.\n\t         *\n\t         * @return {string} The hexadecimally encoded string.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var hexString = CryptoJS.format.Hex.stringify(cipherParams);\n\t         */\n\t        stringify: function (cipherParams) {\n\t            return cipherParams.ciphertext.toString(Hex);\n\t        },\n\n\t        /**\n\t         * Converts a hexadecimally encoded ciphertext string to a cipher params object.\n\t         *\n\t         * @param {string} input The hexadecimally encoded string.\n\t         *\n\t         * @return {CipherParams} The cipher params object.\n\t         *\n\t         * @static\n\t         *\n\t         * @example\n\t         *\n\t         *     var cipherParams = CryptoJS.format.Hex.parse(hexString);\n\t         */\n\t        parse: function (input) {\n\t            var ciphertext = Hex.parse(input);\n\t            return CipherParams.create({ ciphertext: ciphertext });\n\t        }\n\t    };\n\t}());\n\n\n\treturn CryptoJS.format.Hex;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var Base = C_lib.Base;\n\t    var C_enc = C.enc;\n\t    var Utf8 = C_enc.Utf8;\n\t    var C_algo = C.algo;\n\n\t    /**\n\t     * HMAC algorithm.\n\t     */\n\t    var HMAC = C_algo.HMAC = Base.extend({\n\t        /**\n\t         * Initializes a newly created HMAC.\n\t         *\n\t         * @param {Hasher} hasher The hash algorithm to use.\n\t         * @param {WordArray|string} key The secret key.\n\t         *\n\t         * @example\n\t         *\n\t         *     var hmacHasher = CryptoJS.algo.HMAC.create(CryptoJS.algo.SHA256, key);\n\t         */\n\t        init: function (hasher, key) {\n\t            // Init hasher\n\t            hasher = this._hasher = new hasher.init();\n\n\t            // Convert string to WordArray, else assume WordArray already\n\t            if (typeof key == 'string') {\n\t                key = Utf8.parse(key);\n\t            }\n\n\t            // Shortcuts\n\t            var hasherBlockSize = hasher.blockSize;\n\t            var hasherBlockSizeBytes = hasherBlockSize * 4;\n\n\t            // Allow arbitrary length keys\n\t            if (key.sigBytes > hasherBlockSizeBytes) {\n\t                key = hasher.finalize(key);\n\t            }\n\n\t            // Clamp excess bits\n\t            key.clamp();\n\n\t            // Clone key for inner and outer pads\n\t            var oKey = this._oKey = key.clone();\n\t            var iKey = this._iKey = key.clone();\n\n\t            // Shortcuts\n\t            var oKeyWords = oKey.words;\n\t            var iKeyWords = iKey.words;\n\n\t            // XOR keys with pad constants\n\t            for (var i = 0; i < hasherBlockSize; i++) {\n\t                oKeyWords[i] ^= 0x5c5c5c5c;\n\t                iKeyWords[i] ^= 0x36363636;\n\t            }\n\t            oKey.sigBytes = iKey.sigBytes = hasherBlockSizeBytes;\n\n\t            // Set initial values\n\t            this.reset();\n\t        },\n\n\t        /**\n\t         * Resets this HMAC to its initial state.\n\t         *\n\t         * @example\n\t         *\n\t         *     hmacHasher.reset();\n\t         */\n\t        reset: function () {\n\t            // Shortcut\n\t            var hasher = this._hasher;\n\n\t            // Reset\n\t            hasher.reset();\n\t            hasher.update(this._iKey);\n\t        },\n\n\t        /**\n\t         * Updates this HMAC with a message.\n\t         *\n\t         * @param {WordArray|string} messageUpdate The message to append.\n\t         *\n\t         * @return {HMAC} This HMAC instance.\n\t         *\n\t         * @example\n\t         *\n\t         *     hmacHasher.update('message');\n\t         *     hmacHasher.update(wordArray);\n\t         */\n\t        update: function (messageUpdate) {\n\t            this._hasher.update(messageUpdate);\n\n\t            // Chainable\n\t            return this;\n\t        },\n\n\t        /**\n\t         * Finalizes the HMAC computation.\n\t         * Note that the finalize operation is effectively a destructive, read-once operation.\n\t         *\n\t         * @param {WordArray|string} messageUpdate (Optional) A final message update.\n\t         *\n\t         * @return {WordArray} The HMAC.\n\t         *\n\t         * @example\n\t         *\n\t         *     var hmac = hmacHasher.finalize();\n\t         *     var hmac = hmacHasher.finalize('message');\n\t         *     var hmac = hmacHasher.finalize(wordArray);\n\t         */\n\t        finalize: function (messageUpdate) {\n\t            // Shortcut\n\t            var hasher = this._hasher;\n\n\t            // Compute HMAC\n\t            var innerHash = hasher.finalize(messageUpdate);\n\t            hasher.reset();\n\t            var hmac = hasher.finalize(this._oKey.clone().concat(innerHash));\n\n\t            return hmac;\n\t        }\n\t    });\n\t}());\n\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./x64-core\"), require(\"./lib-typedarrays\"), require(\"./enc-utf16\"), require(\"./enc-base64\"), require(\"./md5\"), require(\"./sha1\"), require(\"./sha256\"), require(\"./sha224\"), require(\"./sha512\"), require(\"./sha384\"), require(\"./sha3\"), require(\"./ripemd160\"), require(\"./hmac\"), require(\"./pbkdf2\"), require(\"./evpkdf\"), require(\"./cipher-core\"), require(\"./mode-cfb\"), require(\"./mode-ctr\"), require(\"./mode-ctr-gladman\"), require(\"./mode-ofb\"), require(\"./mode-ecb\"), require(\"./pad-ansix923\"), require(\"./pad-iso10126\"), require(\"./pad-iso97971\"), require(\"./pad-zeropadding\"), require(\"./pad-nopadding\"), require(\"./format-hex\"), require(\"./aes\"), require(\"./tripledes\"), require(\"./rc4\"), require(\"./rabbit\"), require(\"./rabbit-legacy\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./x64-core\", \"./lib-typedarrays\", \"./enc-utf16\", \"./enc-base64\", \"./md5\", \"./sha1\", \"./sha256\", \"./sha224\", \"./sha512\", \"./sha384\", \"./sha3\", \"./ripemd160\", \"./hmac\", \"./pbkdf2\", \"./evpkdf\", \"./cipher-core\", \"./mode-cfb\", \"./mode-ctr\", \"./mode-ctr-gladman\", \"./mode-ofb\", \"./mode-ecb\", \"./pad-ansix923\", \"./pad-iso10126\", \"./pad-iso97971\", \"./pad-zeropadding\", \"./pad-nopadding\", \"./format-hex\", \"./aes\", \"./tripledes\", \"./rc4\", \"./rabbit\", \"./rabbit-legacy\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\troot.CryptoJS = factory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\treturn CryptoJS;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Check if typed arrays are supported\n\t    if (typeof ArrayBuffer != 'function') {\n\t        return;\n\t    }\n\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\n\t    // Reference original init\n\t    var superInit = WordArray.init;\n\n\t    // Augment WordArray.init to handle typed arrays\n\t    var subInit = WordArray.init = function (typedArray) {\n\t        // Convert buffers to uint8\n\t        if (typedArray instanceof ArrayBuffer) {\n\t            typedArray = new Uint8Array(typedArray);\n\t        }\n\n\t        // Convert other array views to uint8\n\t        if (\n\t            typedArray instanceof Int8Array ||\n\t            (typeof Uint8ClampedArray !== \"undefined\" && typedArray instanceof Uint8ClampedArray) ||\n\t            typedArray instanceof Int16Array ||\n\t            typedArray instanceof Uint16Array ||\n\t            typedArray instanceof Int32Array ||\n\t            typedArray instanceof Uint32Array ||\n\t            typedArray instanceof Float32Array ||\n\t            typedArray instanceof Float64Array\n\t        ) {\n\t            typedArray = new Uint8Array(typedArray.buffer, typedArray.byteOffset, typedArray.byteLength);\n\t        }\n\n\t        // Handle Uint8Array\n\t        if (typedArray instanceof Uint8Array) {\n\t            // Shortcut\n\t            var typedArrayByteLength = typedArray.byteLength;\n\n\t            // Extract bytes\n\t            var words = [];\n\t            for (var i = 0; i < typedArrayByteLength; i++) {\n\t                words[i >>> 2] |= typedArray[i] << (24 - (i % 4) * 8);\n\t            }\n\n\t            // Initialize this word array\n\t            superInit.call(this, words, typedArrayByteLength);\n\t        } else {\n\t            // Else call normal init\n\t            superInit.apply(this, arguments);\n\t        }\n\t    };\n\n\t    subInit.prototype = WordArray;\n\t}());\n\n\n\treturn CryptoJS.lib.WordArray;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function (Math) {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var Hasher = C_lib.Hasher;\n\t    var C_algo = C.algo;\n\n\t    // Constants table\n\t    var T = [];\n\n\t    // Compute constants\n\t    (function () {\n\t        for (var i = 0; i < 64; i++) {\n\t            T[i] = (Math.abs(Math.sin(i + 1)) * 0x100000000) | 0;\n\t        }\n\t    }());\n\n\t    /**\n\t     * MD5 hash algorithm.\n\t     */\n\t    var MD5 = C_algo.MD5 = Hasher.extend({\n\t        _doReset: function () {\n\t            this._hash = new WordArray.init([\n\t                0x67452301, 0xefcdab89,\n\t                0x98badcfe, 0x10325476\n\t            ]);\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\t            // Swap endian\n\t            for (var i = 0; i < 16; i++) {\n\t                // Shortcuts\n\t                var offset_i = offset + i;\n\t                var M_offset_i = M[offset_i];\n\n\t                M[offset_i] = (\n\t                    (((M_offset_i << 8)  | (M_offset_i >>> 24)) & 0x00ff00ff) |\n\t                    (((M_offset_i << 24) | (M_offset_i >>> 8))  & 0xff00ff00)\n\t                );\n\t            }\n\n\t            // Shortcuts\n\t            var H = this._hash.words;\n\n\t            var M_offset_0  = M[offset + 0];\n\t            var M_offset_1  = M[offset + 1];\n\t            var M_offset_2  = M[offset + 2];\n\t            var M_offset_3  = M[offset + 3];\n\t            var M_offset_4  = M[offset + 4];\n\t            var M_offset_5  = M[offset + 5];\n\t            var M_offset_6  = M[offset + 6];\n\t            var M_offset_7  = M[offset + 7];\n\t            var M_offset_8  = M[offset + 8];\n\t            var M_offset_9  = M[offset + 9];\n\t            var M_offset_10 = M[offset + 10];\n\t            var M_offset_11 = M[offset + 11];\n\t            var M_offset_12 = M[offset + 12];\n\t            var M_offset_13 = M[offset + 13];\n\t            var M_offset_14 = M[offset + 14];\n\t            var M_offset_15 = M[offset + 15];\n\n\t            // Working varialbes\n\t            var a = H[0];\n\t            var b = H[1];\n\t            var c = H[2];\n\t            var d = H[3];\n\n\t            // Computation\n\t            a = FF(a, b, c, d, M_offset_0,  7,  T[0]);\n\t            d = FF(d, a, b, c, M_offset_1,  12, T[1]);\n\t            c = FF(c, d, a, b, M_offset_2,  17, T[2]);\n\t            b = FF(b, c, d, a, M_offset_3,  22, T[3]);\n\t            a = FF(a, b, c, d, M_offset_4,  7,  T[4]);\n\t            d = FF(d, a, b, c, M_offset_5,  12, T[5]);\n\t            c = FF(c, d, a, b, M_offset_6,  17, T[6]);\n\t            b = FF(b, c, d, a, M_offset_7,  22, T[7]);\n\t            a = FF(a, b, c, d, M_offset_8,  7,  T[8]);\n\t            d = FF(d, a, b, c, M_offset_9,  12, T[9]);\n\t            c = FF(c, d, a, b, M_offset_10, 17, T[10]);\n\t            b = FF(b, c, d, a, M_offset_11, 22, T[11]);\n\t            a = FF(a, b, c, d, M_offset_12, 7,  T[12]);\n\t            d = FF(d, a, b, c, M_offset_13, 12, T[13]);\n\t            c = FF(c, d, a, b, M_offset_14, 17, T[14]);\n\t            b = FF(b, c, d, a, M_offset_15, 22, T[15]);\n\n\t            a = GG(a, b, c, d, M_offset_1,  5,  T[16]);\n\t            d = GG(d, a, b, c, M_offset_6,  9,  T[17]);\n\t            c = GG(c, d, a, b, M_offset_11, 14, T[18]);\n\t            b = GG(b, c, d, a, M_offset_0,  20, T[19]);\n\t            a = GG(a, b, c, d, M_offset_5,  5,  T[20]);\n\t            d = GG(d, a, b, c, M_offset_10, 9,  T[21]);\n\t            c = GG(c, d, a, b, M_offset_15, 14, T[22]);\n\t            b = GG(b, c, d, a, M_offset_4,  20, T[23]);\n\t            a = GG(a, b, c, d, M_offset_9,  5,  T[24]);\n\t            d = GG(d, a, b, c, M_offset_14, 9,  T[25]);\n\t            c = GG(c, d, a, b, M_offset_3,  14, T[26]);\n\t            b = GG(b, c, d, a, M_offset_8,  20, T[27]);\n\t            a = GG(a, b, c, d, M_offset_13, 5,  T[28]);\n\t            d = GG(d, a, b, c, M_offset_2,  9,  T[29]);\n\t            c = GG(c, d, a, b, M_offset_7,  14, T[30]);\n\t            b = GG(b, c, d, a, M_offset_12, 20, T[31]);\n\n\t            a = HH(a, b, c, d, M_offset_5,  4,  T[32]);\n\t            d = HH(d, a, b, c, M_offset_8,  11, T[33]);\n\t            c = HH(c, d, a, b, M_offset_11, 16, T[34]);\n\t            b = HH(b, c, d, a, M_offset_14, 23, T[35]);\n\t            a = HH(a, b, c, d, M_offset_1,  4,  T[36]);\n\t            d = HH(d, a, b, c, M_offset_4,  11, T[37]);\n\t            c = HH(c, d, a, b, M_offset_7,  16, T[38]);\n\t            b = HH(b, c, d, a, M_offset_10, 23, T[39]);\n\t            a = HH(a, b, c, d, M_offset_13, 4,  T[40]);\n\t            d = HH(d, a, b, c, M_offset_0,  11, T[41]);\n\t            c = HH(c, d, a, b, M_offset_3,  16, T[42]);\n\t            b = HH(b, c, d, a, M_offset_6,  23, T[43]);\n\t            a = HH(a, b, c, d, M_offset_9,  4,  T[44]);\n\t            d = HH(d, a, b, c, M_offset_12, 11, T[45]);\n\t            c = HH(c, d, a, b, M_offset_15, 16, T[46]);\n\t            b = HH(b, c, d, a, M_offset_2,  23, T[47]);\n\n\t            a = II(a, b, c, d, M_offset_0,  6,  T[48]);\n\t            d = II(d, a, b, c, M_offset_7,  10, T[49]);\n\t            c = II(c, d, a, b, M_offset_14, 15, T[50]);\n\t            b = II(b, c, d, a, M_offset_5,  21, T[51]);\n\t            a = II(a, b, c, d, M_offset_12, 6,  T[52]);\n\t            d = II(d, a, b, c, M_offset_3,  10, T[53]);\n\t            c = II(c, d, a, b, M_offset_10, 15, T[54]);\n\t            b = II(b, c, d, a, M_offset_1,  21, T[55]);\n\t            a = II(a, b, c, d, M_offset_8,  6,  T[56]);\n\t            d = II(d, a, b, c, M_offset_15, 10, T[57]);\n\t            c = II(c, d, a, b, M_offset_6,  15, T[58]);\n\t            b = II(b, c, d, a, M_offset_13, 21, T[59]);\n\t            a = II(a, b, c, d, M_offset_4,  6,  T[60]);\n\t            d = II(d, a, b, c, M_offset_11, 10, T[61]);\n\t            c = II(c, d, a, b, M_offset_2,  15, T[62]);\n\t            b = II(b, c, d, a, M_offset_9,  21, T[63]);\n\n\t            // Intermediate hash value\n\t            H[0] = (H[0] + a) | 0;\n\t            H[1] = (H[1] + b) | 0;\n\t            H[2] = (H[2] + c) | 0;\n\t            H[3] = (H[3] + d) | 0;\n\t        },\n\n\t        _doFinalize: function () {\n\t            // Shortcuts\n\t            var data = this._data;\n\t            var dataWords = data.words;\n\n\t            var nBitsTotal = this._nDataBytes * 8;\n\t            var nBitsLeft = data.sigBytes * 8;\n\n\t            // Add padding\n\t            dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);\n\n\t            var nBitsTotalH = Math.floor(nBitsTotal / 0x100000000);\n\t            var nBitsTotalL = nBitsTotal;\n\t            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 15] = (\n\t                (((nBitsTotalH << 8)  | (nBitsTotalH >>> 24)) & 0x00ff00ff) |\n\t                (((nBitsTotalH << 24) | (nBitsTotalH >>> 8))  & 0xff00ff00)\n\t            );\n\t            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = (\n\t                (((nBitsTotalL << 8)  | (nBitsTotalL >>> 24)) & 0x00ff00ff) |\n\t                (((nBitsTotalL << 24) | (nBitsTotalL >>> 8))  & 0xff00ff00)\n\t            );\n\n\t            data.sigBytes = (dataWords.length + 1) * 4;\n\n\t            // Hash final blocks\n\t            this._process();\n\n\t            // Shortcuts\n\t            var hash = this._hash;\n\t            var H = hash.words;\n\n\t            // Swap endian\n\t            for (var i = 0; i < 4; i++) {\n\t                // Shortcut\n\t                var H_i = H[i];\n\n\t                H[i] = (((H_i << 8)  | (H_i >>> 24)) & 0x00ff00ff) |\n\t                       (((H_i << 24) | (H_i >>> 8))  & 0xff00ff00);\n\t            }\n\n\t            // Return final computed hash\n\t            return hash;\n\t        },\n\n\t        clone: function () {\n\t            var clone = Hasher.clone.call(this);\n\t            clone._hash = this._hash.clone();\n\n\t            return clone;\n\t        }\n\t    });\n\n\t    function FF(a, b, c, d, x, s, t) {\n\t        var n = a + ((b & c) | (~b & d)) + x + t;\n\t        return ((n << s) | (n >>> (32 - s))) + b;\n\t    }\n\n\t    function GG(a, b, c, d, x, s, t) {\n\t        var n = a + ((b & d) | (c & ~d)) + x + t;\n\t        return ((n << s) | (n >>> (32 - s))) + b;\n\t    }\n\n\t    function HH(a, b, c, d, x, s, t) {\n\t        var n = a + (b ^ c ^ d) + x + t;\n\t        return ((n << s) | (n >>> (32 - s))) + b;\n\t    }\n\n\t    function II(a, b, c, d, x, s, t) {\n\t        var n = a + (c ^ (b | ~d)) + x + t;\n\t        return ((n << s) | (n >>> (32 - s))) + b;\n\t    }\n\n\t    /**\n\t     * Shortcut function to the hasher's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     *\n\t     * @return {WordArray} The hash.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hash = CryptoJS.MD5('message');\n\t     *     var hash = CryptoJS.MD5(wordArray);\n\t     */\n\t    C.MD5 = Hasher._createHelper(MD5);\n\n\t    /**\n\t     * Shortcut function to the HMAC's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     * @param {WordArray|string} key The secret key.\n\t     *\n\t     * @return {WordArray} The HMAC.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hmac = CryptoJS.HmacMD5(message, key);\n\t     */\n\t    C.HmacMD5 = Hasher._createHmacHelper(MD5);\n\t}(Math));\n\n\n\treturn CryptoJS.MD5;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * Cipher Feedback block mode.\n\t */\n\tCryptoJS.mode.CFB = (function () {\n\t    var CFB = CryptoJS.lib.BlockCipherMode.extend();\n\n\t    CFB.Encryptor = CFB.extend({\n\t        processBlock: function (words, offset) {\n\t            // Shortcuts\n\t            var cipher = this._cipher;\n\t            var blockSize = cipher.blockSize;\n\n\t            generateKeystreamAndEncrypt.call(this, words, offset, blockSize, cipher);\n\n\t            // Remember this block to use with next block\n\t            this._prevBlock = words.slice(offset, offset + blockSize);\n\t        }\n\t    });\n\n\t    CFB.Decryptor = CFB.extend({\n\t        processBlock: function (words, offset) {\n\t            // Shortcuts\n\t            var cipher = this._cipher;\n\t            var blockSize = cipher.blockSize;\n\n\t            // Remember this block to use with next block\n\t            var thisBlock = words.slice(offset, offset + blockSize);\n\n\t            generateKeystreamAndEncrypt.call(this, words, offset, blockSize, cipher);\n\n\t            // This block becomes the previous block\n\t            this._prevBlock = thisBlock;\n\t        }\n\t    });\n\n\t    function generateKeystreamAndEncrypt(words, offset, blockSize, cipher) {\n\t        // Shortcut\n\t        var iv = this._iv;\n\n\t        // Generate keystream\n\t        if (iv) {\n\t            var keystream = iv.slice(0);\n\n\t            // Remove IV for subsequent blocks\n\t            this._iv = undefined;\n\t        } else {\n\t            var keystream = this._prevBlock;\n\t        }\n\t        cipher.encryptBlock(keystream, 0);\n\n\t        // Encrypt\n\t        for (var i = 0; i < blockSize; i++) {\n\t            words[offset + i] ^= keystream[i];\n\t        }\n\t    }\n\n\t    return CFB;\n\t}());\n\n\n\treturn CryptoJS.mode.CFB;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/** @preserve\n\t * Counter block mode compatible with  Dr Brian Gladman fileenc.c\n\t * derived from CryptoJS.mode.CTR\n\t * Jan Hruby jhruby.web@gmail.com\n\t */\n\tCryptoJS.mode.CTRGladman = (function () {\n\t    var CTRGladman = CryptoJS.lib.BlockCipherMode.extend();\n\n\t\tfunction incWord(word)\n\t\t{\n\t\t\tif (((word >> 24) & 0xff) === 0xff) { //overflow\n\t\t\tvar b1 = (word >> 16)&0xff;\n\t\t\tvar b2 = (word >> 8)&0xff;\n\t\t\tvar b3 = word & 0xff;\n\n\t\t\tif (b1 === 0xff) // overflow b1\n\t\t\t{\n\t\t\tb1 = 0;\n\t\t\tif (b2 === 0xff)\n\t\t\t{\n\t\t\t\tb2 = 0;\n\t\t\t\tif (b3 === 0xff)\n\t\t\t\t{\n\t\t\t\t\tb3 = 0;\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\t++b3;\n\t\t\t\t}\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t++b2;\n\t\t\t}\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t++b1;\n\t\t\t}\n\n\t\t\tword = 0;\n\t\t\tword += (b1 << 16);\n\t\t\tword += (b2 << 8);\n\t\t\tword += b3;\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\tword += (0x01 << 24);\n\t\t\t}\n\t\t\treturn word;\n\t\t}\n\n\t\tfunction incCounter(counter)\n\t\t{\n\t\t\tif ((counter[0] = incWord(counter[0])) === 0)\n\t\t\t{\n\t\t\t\t// encr_data in fileenc.c from  Dr Brian Gladman's counts only with DWORD j < 8\n\t\t\t\tcounter[1] = incWord(counter[1]);\n\t\t\t}\n\t\t\treturn counter;\n\t\t}\n\n\t    var Encryptor = CTRGladman.Encryptor = CTRGladman.extend({\n\t        processBlock: function (words, offset) {\n\t            // Shortcuts\n\t            var cipher = this._cipher\n\t            var blockSize = cipher.blockSize;\n\t            var iv = this._iv;\n\t            var counter = this._counter;\n\n\t            // Generate keystream\n\t            if (iv) {\n\t                counter = this._counter = iv.slice(0);\n\n\t                // Remove IV for subsequent blocks\n\t                this._iv = undefined;\n\t            }\n\n\t\t\t\tincCounter(counter);\n\n\t\t\t\tvar keystream = counter.slice(0);\n\t            cipher.encryptBlock(keystream, 0);\n\n\t            // Encrypt\n\t            for (var i = 0; i < blockSize; i++) {\n\t                words[offset + i] ^= keystream[i];\n\t            }\n\t        }\n\t    });\n\n\t    CTRGladman.Decryptor = Encryptor;\n\n\t    return CTRGladman;\n\t}());\n\n\n\n\n\treturn CryptoJS.mode.CTRGladman;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * Counter block mode.\n\t */\n\tCryptoJS.mode.CTR = (function () {\n\t    var CTR = CryptoJS.lib.BlockCipherMode.extend();\n\n\t    var Encryptor = CTR.Encryptor = CTR.extend({\n\t        processBlock: function (words, offset) {\n\t            // Shortcuts\n\t            var cipher = this._cipher\n\t            var blockSize = cipher.blockSize;\n\t            var iv = this._iv;\n\t            var counter = this._counter;\n\n\t            // Generate keystream\n\t            if (iv) {\n\t                counter = this._counter = iv.slice(0);\n\n\t                // Remove IV for subsequent blocks\n\t                this._iv = undefined;\n\t            }\n\t            var keystream = counter.slice(0);\n\t            cipher.encryptBlock(keystream, 0);\n\n\t            // Increment counter\n\t            counter[blockSize - 1] = (counter[blockSize - 1] + 1) | 0\n\n\t            // Encrypt\n\t            for (var i = 0; i < blockSize; i++) {\n\t                words[offset + i] ^= keystream[i];\n\t            }\n\t        }\n\t    });\n\n\t    CTR.Decryptor = Encryptor;\n\n\t    return CTR;\n\t}());\n\n\n\treturn CryptoJS.mode.CTR;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * Electronic Codebook block mode.\n\t */\n\tCryptoJS.mode.ECB = (function () {\n\t    var ECB = CryptoJS.lib.BlockCipherMode.extend();\n\n\t    ECB.Encryptor = ECB.extend({\n\t        processBlock: function (words, offset) {\n\t            this._cipher.encryptBlock(words, offset);\n\t        }\n\t    });\n\n\t    ECB.Decryptor = ECB.extend({\n\t        processBlock: function (words, offset) {\n\t            this._cipher.decryptBlock(words, offset);\n\t        }\n\t    });\n\n\t    return ECB;\n\t}());\n\n\n\treturn CryptoJS.mode.ECB;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * Output Feedback block mode.\n\t */\n\tCryptoJS.mode.OFB = (function () {\n\t    var OFB = CryptoJS.lib.BlockCipherMode.extend();\n\n\t    var Encryptor = OFB.Encryptor = OFB.extend({\n\t        processBlock: function (words, offset) {\n\t            // Shortcuts\n\t            var cipher = this._cipher\n\t            var blockSize = cipher.blockSize;\n\t            var iv = this._iv;\n\t            var keystream = this._keystream;\n\n\t            // Generate keystream\n\t            if (iv) {\n\t                keystream = this._keystream = iv.slice(0);\n\n\t                // Remove IV for subsequent blocks\n\t                this._iv = undefined;\n\t            }\n\t            cipher.encryptBlock(keystream, 0);\n\n\t            // Encrypt\n\t            for (var i = 0; i < blockSize; i++) {\n\t                words[offset + i] ^= keystream[i];\n\t            }\n\t        }\n\t    });\n\n\t    OFB.Decryptor = Encryptor;\n\n\t    return OFB;\n\t}());\n\n\n\treturn CryptoJS.mode.OFB;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * ANSI X.923 padding strategy.\n\t */\n\tCryptoJS.pad.AnsiX923 = {\n\t    pad: function (data, blockSize) {\n\t        // Shortcuts\n\t        var dataSigBytes = data.sigBytes;\n\t        var blockSizeBytes = blockSize * 4;\n\n\t        // Count padding bytes\n\t        var nPaddingBytes = blockSizeBytes - dataSigBytes % blockSizeBytes;\n\n\t        // Compute last byte position\n\t        var lastBytePos = dataSigBytes + nPaddingBytes - 1;\n\n\t        // Pad\n\t        data.clamp();\n\t        data.words[lastBytePos >>> 2] |= nPaddingBytes << (24 - (lastBytePos % 4) * 8);\n\t        data.sigBytes += nPaddingBytes;\n\t    },\n\n\t    unpad: function (data) {\n\t        // Get number of padding bytes from last byte\n\t        var nPaddingBytes = data.words[(data.sigBytes - 1) >>> 2] & 0xff;\n\n\t        // Remove padding\n\t        data.sigBytes -= nPaddingBytes;\n\t    }\n\t};\n\n\n\treturn CryptoJS.pad.Ansix923;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * ISO 10126 padding strategy.\n\t */\n\tCryptoJS.pad.Iso10126 = {\n\t    pad: function (data, blockSize) {\n\t        // Shortcut\n\t        var blockSizeBytes = blockSize * 4;\n\n\t        // Count padding bytes\n\t        var nPaddingBytes = blockSizeBytes - data.sigBytes % blockSizeBytes;\n\n\t        // Pad\n\t        data.concat(CryptoJS.lib.WordArray.random(nPaddingBytes - 1)).\n\t             concat(CryptoJS.lib.WordArray.create([nPaddingBytes << 24], 1));\n\t    },\n\n\t    unpad: function (data) {\n\t        // Get number of padding bytes from last byte\n\t        var nPaddingBytes = data.words[(data.sigBytes - 1) >>> 2] & 0xff;\n\n\t        // Remove padding\n\t        data.sigBytes -= nPaddingBytes;\n\t    }\n\t};\n\n\n\treturn CryptoJS.pad.Iso10126;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * ISO/IEC 9797-1 Padding Method 2.\n\t */\n\tCryptoJS.pad.Iso97971 = {\n\t    pad: function (data, blockSize) {\n\t        // Add 0x80 byte\n\t        data.concat(CryptoJS.lib.WordArray.create([0x80000000], 1));\n\n\t        // Zero pad the rest\n\t        CryptoJS.pad.ZeroPadding.pad(data, blockSize);\n\t    },\n\n\t    unpad: function (data) {\n\t        // Remove zero padding\n\t        CryptoJS.pad.ZeroPadding.unpad(data);\n\n\t        // Remove one more byte -- the 0x80 byte\n\t        data.sigBytes--;\n\t    }\n\t};\n\n\n\treturn CryptoJS.pad.Iso97971;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * A noop padding strategy.\n\t */\n\tCryptoJS.pad.NoPadding = {\n\t    pad: function () {\n\t    },\n\n\t    unpad: function () {\n\t    }\n\t};\n\n\n\treturn CryptoJS.pad.NoPadding;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/**\n\t * Zero padding strategy.\n\t */\n\tCryptoJS.pad.ZeroPadding = {\n\t    pad: function (data, blockSize) {\n\t        // Shortcut\n\t        var blockSizeBytes = blockSize * 4;\n\n\t        // Pad\n\t        data.clamp();\n\t        data.sigBytes += blockSizeBytes - ((data.sigBytes % blockSizeBytes) || blockSizeBytes);\n\t    },\n\n\t    unpad: function (data) {\n\t        // Shortcut\n\t        var dataWords = data.words;\n\n\t        // Unpad\n\t        var i = data.sigBytes - 1;\n\t        while (!((dataWords[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff)) {\n\t            i--;\n\t        }\n\t        data.sigBytes = i + 1;\n\t    }\n\t};\n\n\n\treturn CryptoJS.pad.ZeroPadding;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./sha1\"), require(\"./hmac\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./sha1\", \"./hmac\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var Base = C_lib.Base;\n\t    var WordArray = C_lib.WordArray;\n\t    var C_algo = C.algo;\n\t    var SHA1 = C_algo.SHA1;\n\t    var HMAC = C_algo.HMAC;\n\n\t    /**\n\t     * Password-Based Key Derivation Function 2 algorithm.\n\t     */\n\t    var PBKDF2 = C_algo.PBKDF2 = Base.extend({\n\t        /**\n\t         * Configuration options.\n\t         *\n\t         * @property {number} keySize The key size in words to generate. Default: 4 (128 bits)\n\t         * @property {Hasher} hasher The hasher to use. Default: SHA1\n\t         * @property {number} iterations The number of iterations to perform. Default: 1\n\t         */\n\t        cfg: Base.extend({\n\t            keySize: 128/32,\n\t            hasher: SHA1,\n\t            iterations: 1\n\t        }),\n\n\t        /**\n\t         * Initializes a newly created key derivation function.\n\t         *\n\t         * @param {Object} cfg (Optional) The configuration options to use for the derivation.\n\t         *\n\t         * @example\n\t         *\n\t         *     var kdf = CryptoJS.algo.PBKDF2.create();\n\t         *     var kdf = CryptoJS.algo.PBKDF2.create({ keySize: 8 });\n\t         *     var kdf = CryptoJS.algo.PBKDF2.create({ keySize: 8, iterations: 1000 });\n\t         */\n\t        init: function (cfg) {\n\t            this.cfg = this.cfg.extend(cfg);\n\t        },\n\n\t        /**\n\t         * Computes the Password-Based Key Derivation Function 2.\n\t         *\n\t         * @param {WordArray|string} password The password.\n\t         * @param {WordArray|string} salt A salt.\n\t         *\n\t         * @return {WordArray} The derived key.\n\t         *\n\t         * @example\n\t         *\n\t         *     var key = kdf.compute(password, salt);\n\t         */\n\t        compute: function (password, salt) {\n\t            // Shortcut\n\t            var cfg = this.cfg;\n\n\t            // Init HMAC\n\t            var hmac = HMAC.create(cfg.hasher, password);\n\n\t            // Initial values\n\t            var derivedKey = WordArray.create();\n\t            var blockIndex = WordArray.create([0x00000001]);\n\n\t            // Shortcuts\n\t            var derivedKeyWords = derivedKey.words;\n\t            var blockIndexWords = blockIndex.words;\n\t            var keySize = cfg.keySize;\n\t            var iterations = cfg.iterations;\n\n\t            // Generate key\n\t            while (derivedKeyWords.length < keySize) {\n\t                var block = hmac.update(salt).finalize(blockIndex);\n\t                hmac.reset();\n\n\t                // Shortcuts\n\t                var blockWords = block.words;\n\t                var blockWordsLength = blockWords.length;\n\n\t                // Iterations\n\t                var intermediate = block;\n\t                for (var i = 1; i < iterations; i++) {\n\t                    intermediate = hmac.finalize(intermediate);\n\t                    hmac.reset();\n\n\t                    // Shortcut\n\t                    var intermediateWords = intermediate.words;\n\n\t                    // XOR intermediate with block\n\t                    for (var j = 0; j < blockWordsLength; j++) {\n\t                        blockWords[j] ^= intermediateWords[j];\n\t                    }\n\t                }\n\n\t                derivedKey.concat(block);\n\t                blockIndexWords[0]++;\n\t            }\n\t            derivedKey.sigBytes = keySize * 4;\n\n\t            return derivedKey;\n\t        }\n\t    });\n\n\t    /**\n\t     * Computes the Password-Based Key Derivation Function 2.\n\t     *\n\t     * @param {WordArray|string} password The password.\n\t     * @param {WordArray|string} salt A salt.\n\t     * @param {Object} cfg (Optional) The configuration options to use for this computation.\n\t     *\n\t     * @return {WordArray} The derived key.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var key = CryptoJS.PBKDF2(password, salt);\n\t     *     var key = CryptoJS.PBKDF2(password, salt, { keySize: 8 });\n\t     *     var key = CryptoJS.PBKDF2(password, salt, { keySize: 8, iterations: 1000 });\n\t     */\n\t    C.PBKDF2 = function (password, salt, cfg) {\n\t        return PBKDF2.create(cfg).compute(password, salt);\n\t    };\n\t}());\n\n\n\treturn CryptoJS.PBKDF2;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./enc-base64\"), require(\"./md5\"), require(\"./evpkdf\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./enc-base64\", \"./md5\", \"./evpkdf\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var StreamCipher = C_lib.StreamCipher;\n\t    var C_algo = C.algo;\n\n\t    // Reusable objects\n\t    var S  = [];\n\t    var C_ = [];\n\t    var G  = [];\n\n\t    /**\n\t     * Rabbit stream cipher algorithm.\n\t     *\n\t     * This is a legacy version that neglected to convert the key to little-endian.\n\t     * This error doesn't affect the cipher's security,\n\t     * but it does affect its compatibility with other implementations.\n\t     */\n\t    var RabbitLegacy = C_algo.RabbitLegacy = StreamCipher.extend({\n\t        _doReset: function () {\n\t            // Shortcuts\n\t            var K = this._key.words;\n\t            var iv = this.cfg.iv;\n\n\t            // Generate initial state values\n\t            var X = this._X = [\n\t                K[0], (K[3] << 16) | (K[2] >>> 16),\n\t                K[1], (K[0] << 16) | (K[3] >>> 16),\n\t                K[2], (K[1] << 16) | (K[0] >>> 16),\n\t                K[3], (K[2] << 16) | (K[1] >>> 16)\n\t            ];\n\n\t            // Generate initial counter values\n\t            var C = this._C = [\n\t                (K[2] << 16) | (K[2] >>> 16), (K[0] & 0xffff0000) | (K[1] & 0x0000ffff),\n\t                (K[3] << 16) | (K[3] >>> 16), (K[1] & 0xffff0000) | (K[2] & 0x0000ffff),\n\t                (K[0] << 16) | (K[0] >>> 16), (K[2] & 0xffff0000) | (K[3] & 0x0000ffff),\n\t                (K[1] << 16) | (K[1] >>> 16), (K[3] & 0xffff0000) | (K[0] & 0x0000ffff)\n\t            ];\n\n\t            // Carry bit\n\t            this._b = 0;\n\n\t            // Iterate the system four times\n\t            for (var i = 0; i < 4; i++) {\n\t                nextState.call(this);\n\t            }\n\n\t            // Modify the counters\n\t            for (var i = 0; i < 8; i++) {\n\t                C[i] ^= X[(i + 4) & 7];\n\t            }\n\n\t            // IV setup\n\t            if (iv) {\n\t                // Shortcuts\n\t                var IV = iv.words;\n\t                var IV_0 = IV[0];\n\t                var IV_1 = IV[1];\n\n\t                // Generate four subvectors\n\t                var i0 = (((IV_0 << 8) | (IV_0 >>> 24)) & 0x00ff00ff) | (((IV_0 << 24) | (IV_0 >>> 8)) & 0xff00ff00);\n\t                var i2 = (((IV_1 << 8) | (IV_1 >>> 24)) & 0x00ff00ff) | (((IV_1 << 24) | (IV_1 >>> 8)) & 0xff00ff00);\n\t                var i1 = (i0 >>> 16) | (i2 & 0xffff0000);\n\t                var i3 = (i2 << 16)  | (i0 & 0x0000ffff);\n\n\t                // Modify counter values\n\t                C[0] ^= i0;\n\t                C[1] ^= i1;\n\t                C[2] ^= i2;\n\t                C[3] ^= i3;\n\t                C[4] ^= i0;\n\t                C[5] ^= i1;\n\t                C[6] ^= i2;\n\t                C[7] ^= i3;\n\n\t                // Iterate the system four times\n\t                for (var i = 0; i < 4; i++) {\n\t                    nextState.call(this);\n\t                }\n\t            }\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\t            // Shortcut\n\t            var X = this._X;\n\n\t            // Iterate the system\n\t            nextState.call(this);\n\n\t            // Generate four keystream words\n\t            S[0] = X[0] ^ (X[5] >>> 16) ^ (X[3] << 16);\n\t            S[1] = X[2] ^ (X[7] >>> 16) ^ (X[5] << 16);\n\t            S[2] = X[4] ^ (X[1] >>> 16) ^ (X[7] << 16);\n\t            S[3] = X[6] ^ (X[3] >>> 16) ^ (X[1] << 16);\n\n\t            for (var i = 0; i < 4; i++) {\n\t                // Swap endian\n\t                S[i] = (((S[i] << 8)  | (S[i] >>> 24)) & 0x00ff00ff) |\n\t                       (((S[i] << 24) | (S[i] >>> 8))  & 0xff00ff00);\n\n\t                // Encrypt\n\t                M[offset + i] ^= S[i];\n\t            }\n\t        },\n\n\t        blockSize: 128/32,\n\n\t        ivSize: 64/32\n\t    });\n\n\t    function nextState() {\n\t        // Shortcuts\n\t        var X = this._X;\n\t        var C = this._C;\n\n\t        // Save old counter values\n\t        for (var i = 0; i < 8; i++) {\n\t            C_[i] = C[i];\n\t        }\n\n\t        // Calculate new counter values\n\t        C[0] = (C[0] + 0x4d34d34d + this._b) | 0;\n\t        C[1] = (C[1] + 0xd34d34d3 + ((C[0] >>> 0) < (C_[0] >>> 0) ? 1 : 0)) | 0;\n\t        C[2] = (C[2] + 0x34d34d34 + ((C[1] >>> 0) < (C_[1] >>> 0) ? 1 : 0)) | 0;\n\t        C[3] = (C[3] + 0x4d34d34d + ((C[2] >>> 0) < (C_[2] >>> 0) ? 1 : 0)) | 0;\n\t        C[4] = (C[4] + 0xd34d34d3 + ((C[3] >>> 0) < (C_[3] >>> 0) ? 1 : 0)) | 0;\n\t        C[5] = (C[5] + 0x34d34d34 + ((C[4] >>> 0) < (C_[4] >>> 0) ? 1 : 0)) | 0;\n\t        C[6] = (C[6] + 0x4d34d34d + ((C[5] >>> 0) < (C_[5] >>> 0) ? 1 : 0)) | 0;\n\t        C[7] = (C[7] + 0xd34d34d3 + ((C[6] >>> 0) < (C_[6] >>> 0) ? 1 : 0)) | 0;\n\t        this._b = (C[7] >>> 0) < (C_[7] >>> 0) ? 1 : 0;\n\n\t        // Calculate the g-values\n\t        for (var i = 0; i < 8; i++) {\n\t            var gx = X[i] + C[i];\n\n\t            // Construct high and low argument for squaring\n\t            var ga = gx & 0xffff;\n\t            var gb = gx >>> 16;\n\n\t            // Calculate high and low result of squaring\n\t            var gh = ((((ga * ga) >>> 17) + ga * gb) >>> 15) + gb * gb;\n\t            var gl = (((gx & 0xffff0000) * gx) | 0) + (((gx & 0x0000ffff) * gx) | 0);\n\n\t            // High XOR low\n\t            G[i] = gh ^ gl;\n\t        }\n\n\t        // Calculate new state values\n\t        X[0] = (G[0] + ((G[7] << 16) | (G[7] >>> 16)) + ((G[6] << 16) | (G[6] >>> 16))) | 0;\n\t        X[1] = (G[1] + ((G[0] << 8)  | (G[0] >>> 24)) + G[7]) | 0;\n\t        X[2] = (G[2] + ((G[1] << 16) | (G[1] >>> 16)) + ((G[0] << 16) | (G[0] >>> 16))) | 0;\n\t        X[3] = (G[3] + ((G[2] << 8)  | (G[2] >>> 24)) + G[1]) | 0;\n\t        X[4] = (G[4] + ((G[3] << 16) | (G[3] >>> 16)) + ((G[2] << 16) | (G[2] >>> 16))) | 0;\n\t        X[5] = (G[5] + ((G[4] << 8)  | (G[4] >>> 24)) + G[3]) | 0;\n\t        X[6] = (G[6] + ((G[5] << 16) | (G[5] >>> 16)) + ((G[4] << 16) | (G[4] >>> 16))) | 0;\n\t        X[7] = (G[7] + ((G[6] << 8)  | (G[6] >>> 24)) + G[5]) | 0;\n\t    }\n\n\t    /**\n\t     * Shortcut functions to the cipher's object interface.\n\t     *\n\t     * @example\n\t     *\n\t     *     var ciphertext = CryptoJS.RabbitLegacy.encrypt(message, key, cfg);\n\t     *     var plaintext  = CryptoJS.RabbitLegacy.decrypt(ciphertext, key, cfg);\n\t     */\n\t    C.RabbitLegacy = StreamCipher._createHelper(RabbitLegacy);\n\t}());\n\n\n\treturn CryptoJS.RabbitLegacy;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./enc-base64\"), require(\"./md5\"), require(\"./evpkdf\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./enc-base64\", \"./md5\", \"./evpkdf\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var StreamCipher = C_lib.StreamCipher;\n\t    var C_algo = C.algo;\n\n\t    // Reusable objects\n\t    var S  = [];\n\t    var C_ = [];\n\t    var G  = [];\n\n\t    /**\n\t     * Rabbit stream cipher algorithm\n\t     */\n\t    var Rabbit = C_algo.Rabbit = StreamCipher.extend({\n\t        _doReset: function () {\n\t            // Shortcuts\n\t            var K = this._key.words;\n\t            var iv = this.cfg.iv;\n\n\t            // Swap endian\n\t            for (var i = 0; i < 4; i++) {\n\t                K[i] = (((K[i] << 8)  | (K[i] >>> 24)) & 0x00ff00ff) |\n\t                       (((K[i] << 24) | (K[i] >>> 8))  & 0xff00ff00);\n\t            }\n\n\t            // Generate initial state values\n\t            var X = this._X = [\n\t                K[0], (K[3] << 16) | (K[2] >>> 16),\n\t                K[1], (K[0] << 16) | (K[3] >>> 16),\n\t                K[2], (K[1] << 16) | (K[0] >>> 16),\n\t                K[3], (K[2] << 16) | (K[1] >>> 16)\n\t            ];\n\n\t            // Generate initial counter values\n\t            var C = this._C = [\n\t                (K[2] << 16) | (K[2] >>> 16), (K[0] & 0xffff0000) | (K[1] & 0x0000ffff),\n\t                (K[3] << 16) | (K[3] >>> 16), (K[1] & 0xffff0000) | (K[2] & 0x0000ffff),\n\t                (K[0] << 16) | (K[0] >>> 16), (K[2] & 0xffff0000) | (K[3] & 0x0000ffff),\n\t                (K[1] << 16) | (K[1] >>> 16), (K[3] & 0xffff0000) | (K[0] & 0x0000ffff)\n\t            ];\n\n\t            // Carry bit\n\t            this._b = 0;\n\n\t            // Iterate the system four times\n\t            for (var i = 0; i < 4; i++) {\n\t                nextState.call(this);\n\t            }\n\n\t            // Modify the counters\n\t            for (var i = 0; i < 8; i++) {\n\t                C[i] ^= X[(i + 4) & 7];\n\t            }\n\n\t            // IV setup\n\t            if (iv) {\n\t                // Shortcuts\n\t                var IV = iv.words;\n\t                var IV_0 = IV[0];\n\t                var IV_1 = IV[1];\n\n\t                // Generate four subvectors\n\t                var i0 = (((IV_0 << 8) | (IV_0 >>> 24)) & 0x00ff00ff) | (((IV_0 << 24) | (IV_0 >>> 8)) & 0xff00ff00);\n\t                var i2 = (((IV_1 << 8) | (IV_1 >>> 24)) & 0x00ff00ff) | (((IV_1 << 24) | (IV_1 >>> 8)) & 0xff00ff00);\n\t                var i1 = (i0 >>> 16) | (i2 & 0xffff0000);\n\t                var i3 = (i2 << 16)  | (i0 & 0x0000ffff);\n\n\t                // Modify counter values\n\t                C[0] ^= i0;\n\t                C[1] ^= i1;\n\t                C[2] ^= i2;\n\t                C[3] ^= i3;\n\t                C[4] ^= i0;\n\t                C[5] ^= i1;\n\t                C[6] ^= i2;\n\t                C[7] ^= i3;\n\n\t                // Iterate the system four times\n\t                for (var i = 0; i < 4; i++) {\n\t                    nextState.call(this);\n\t                }\n\t            }\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\t            // Shortcut\n\t            var X = this._X;\n\n\t            // Iterate the system\n\t            nextState.call(this);\n\n\t            // Generate four keystream words\n\t            S[0] = X[0] ^ (X[5] >>> 16) ^ (X[3] << 16);\n\t            S[1] = X[2] ^ (X[7] >>> 16) ^ (X[5] << 16);\n\t            S[2] = X[4] ^ (X[1] >>> 16) ^ (X[7] << 16);\n\t            S[3] = X[6] ^ (X[3] >>> 16) ^ (X[1] << 16);\n\n\t            for (var i = 0; i < 4; i++) {\n\t                // Swap endian\n\t                S[i] = (((S[i] << 8)  | (S[i] >>> 24)) & 0x00ff00ff) |\n\t                       (((S[i] << 24) | (S[i] >>> 8))  & 0xff00ff00);\n\n\t                // Encrypt\n\t                M[offset + i] ^= S[i];\n\t            }\n\t        },\n\n\t        blockSize: 128/32,\n\n\t        ivSize: 64/32\n\t    });\n\n\t    function nextState() {\n\t        // Shortcuts\n\t        var X = this._X;\n\t        var C = this._C;\n\n\t        // Save old counter values\n\t        for (var i = 0; i < 8; i++) {\n\t            C_[i] = C[i];\n\t        }\n\n\t        // Calculate new counter values\n\t        C[0] = (C[0] + 0x4d34d34d + this._b) | 0;\n\t        C[1] = (C[1] + 0xd34d34d3 + ((C[0] >>> 0) < (C_[0] >>> 0) ? 1 : 0)) | 0;\n\t        C[2] = (C[2] + 0x34d34d34 + ((C[1] >>> 0) < (C_[1] >>> 0) ? 1 : 0)) | 0;\n\t        C[3] = (C[3] + 0x4d34d34d + ((C[2] >>> 0) < (C_[2] >>> 0) ? 1 : 0)) | 0;\n\t        C[4] = (C[4] + 0xd34d34d3 + ((C[3] >>> 0) < (C_[3] >>> 0) ? 1 : 0)) | 0;\n\t        C[5] = (C[5] + 0x34d34d34 + ((C[4] >>> 0) < (C_[4] >>> 0) ? 1 : 0)) | 0;\n\t        C[6] = (C[6] + 0x4d34d34d + ((C[5] >>> 0) < (C_[5] >>> 0) ? 1 : 0)) | 0;\n\t        C[7] = (C[7] + 0xd34d34d3 + ((C[6] >>> 0) < (C_[6] >>> 0) ? 1 : 0)) | 0;\n\t        this._b = (C[7] >>> 0) < (C_[7] >>> 0) ? 1 : 0;\n\n\t        // Calculate the g-values\n\t        for (var i = 0; i < 8; i++) {\n\t            var gx = X[i] + C[i];\n\n\t            // Construct high and low argument for squaring\n\t            var ga = gx & 0xffff;\n\t            var gb = gx >>> 16;\n\n\t            // Calculate high and low result of squaring\n\t            var gh = ((((ga * ga) >>> 17) + ga * gb) >>> 15) + gb * gb;\n\t            var gl = (((gx & 0xffff0000) * gx) | 0) + (((gx & 0x0000ffff) * gx) | 0);\n\n\t            // High XOR low\n\t            G[i] = gh ^ gl;\n\t        }\n\n\t        // Calculate new state values\n\t        X[0] = (G[0] + ((G[7] << 16) | (G[7] >>> 16)) + ((G[6] << 16) | (G[6] >>> 16))) | 0;\n\t        X[1] = (G[1] + ((G[0] << 8)  | (G[0] >>> 24)) + G[7]) | 0;\n\t        X[2] = (G[2] + ((G[1] << 16) | (G[1] >>> 16)) + ((G[0] << 16) | (G[0] >>> 16))) | 0;\n\t        X[3] = (G[3] + ((G[2] << 8)  | (G[2] >>> 24)) + G[1]) | 0;\n\t        X[4] = (G[4] + ((G[3] << 16) | (G[3] >>> 16)) + ((G[2] << 16) | (G[2] >>> 16))) | 0;\n\t        X[5] = (G[5] + ((G[4] << 8)  | (G[4] >>> 24)) + G[3]) | 0;\n\t        X[6] = (G[6] + ((G[5] << 16) | (G[5] >>> 16)) + ((G[4] << 16) | (G[4] >>> 16))) | 0;\n\t        X[7] = (G[7] + ((G[6] << 8)  | (G[6] >>> 24)) + G[5]) | 0;\n\t    }\n\n\t    /**\n\t     * Shortcut functions to the cipher's object interface.\n\t     *\n\t     * @example\n\t     *\n\t     *     var ciphertext = CryptoJS.Rabbit.encrypt(message, key, cfg);\n\t     *     var plaintext  = CryptoJS.Rabbit.decrypt(ciphertext, key, cfg);\n\t     */\n\t    C.Rabbit = StreamCipher._createHelper(Rabbit);\n\t}());\n\n\n\treturn CryptoJS.Rabbit;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./enc-base64\"), require(\"./md5\"), require(\"./evpkdf\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./enc-base64\", \"./md5\", \"./evpkdf\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var StreamCipher = C_lib.StreamCipher;\n\t    var C_algo = C.algo;\n\n\t    /**\n\t     * RC4 stream cipher algorithm.\n\t     */\n\t    var RC4 = C_algo.RC4 = StreamCipher.extend({\n\t        _doReset: function () {\n\t            // Shortcuts\n\t            var key = this._key;\n\t            var keyWords = key.words;\n\t            var keySigBytes = key.sigBytes;\n\n\t            // Init sbox\n\t            var S = this._S = [];\n\t            for (var i = 0; i < 256; i++) {\n\t                S[i] = i;\n\t            }\n\n\t            // Key setup\n\t            for (var i = 0, j = 0; i < 256; i++) {\n\t                var keyByteIndex = i % keySigBytes;\n\t                var keyByte = (keyWords[keyByteIndex >>> 2] >>> (24 - (keyByteIndex % 4) * 8)) & 0xff;\n\n\t                j = (j + S[i] + keyByte) % 256;\n\n\t                // Swap\n\t                var t = S[i];\n\t                S[i] = S[j];\n\t                S[j] = t;\n\t            }\n\n\t            // Counters\n\t            this._i = this._j = 0;\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\t            M[offset] ^= generateKeystreamWord.call(this);\n\t        },\n\n\t        keySize: 256/32,\n\n\t        ivSize: 0\n\t    });\n\n\t    function generateKeystreamWord() {\n\t        // Shortcuts\n\t        var S = this._S;\n\t        var i = this._i;\n\t        var j = this._j;\n\n\t        // Generate keystream word\n\t        var keystreamWord = 0;\n\t        for (var n = 0; n < 4; n++) {\n\t            i = (i + 1) % 256;\n\t            j = (j + S[i]) % 256;\n\n\t            // Swap\n\t            var t = S[i];\n\t            S[i] = S[j];\n\t            S[j] = t;\n\n\t            keystreamWord |= S[(S[i] + S[j]) % 256] << (24 - n * 8);\n\t        }\n\n\t        // Update counters\n\t        this._i = i;\n\t        this._j = j;\n\n\t        return keystreamWord;\n\t    }\n\n\t    /**\n\t     * Shortcut functions to the cipher's object interface.\n\t     *\n\t     * @example\n\t     *\n\t     *     var ciphertext = CryptoJS.RC4.encrypt(message, key, cfg);\n\t     *     var plaintext  = CryptoJS.RC4.decrypt(ciphertext, key, cfg);\n\t     */\n\t    C.RC4 = StreamCipher._createHelper(RC4);\n\n\t    /**\n\t     * Modified RC4 stream cipher algorithm.\n\t     */\n\t    var RC4Drop = C_algo.RC4Drop = RC4.extend({\n\t        /**\n\t         * Configuration options.\n\t         *\n\t         * @property {number} drop The number of keystream words to drop. Default 192\n\t         */\n\t        cfg: RC4.cfg.extend({\n\t            drop: 192\n\t        }),\n\n\t        _doReset: function () {\n\t            RC4._doReset.call(this);\n\n\t            // Drop\n\t            for (var i = this.cfg.drop; i > 0; i--) {\n\t                generateKeystreamWord.call(this);\n\t            }\n\t        }\n\t    });\n\n\t    /**\n\t     * Shortcut functions to the cipher's object interface.\n\t     *\n\t     * @example\n\t     *\n\t     *     var ciphertext = CryptoJS.RC4Drop.encrypt(message, key, cfg);\n\t     *     var plaintext  = CryptoJS.RC4Drop.decrypt(ciphertext, key, cfg);\n\t     */\n\t    C.RC4Drop = StreamCipher._createHelper(RC4Drop);\n\t}());\n\n\n\treturn CryptoJS.RC4;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t/** @preserve\n\t(c) 2012 by Cédric Mesnil. All rights reserved.\n\n\tRedistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:\n\n\t    - Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.\n\t    - Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.\n\n\tTHIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS \"AS IS\" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.\n\t*/\n\n\t(function (Math) {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var Hasher = C_lib.Hasher;\n\t    var C_algo = C.algo;\n\n\t    // Constants table\n\t    var _zl = WordArray.create([\n\t        0,  1,  2,  3,  4,  5,  6,  7,  8,  9, 10, 11, 12, 13, 14, 15,\n\t        7,  4, 13,  1, 10,  6, 15,  3, 12,  0,  9,  5,  2, 14, 11,  8,\n\t        3, 10, 14,  4,  9, 15,  8,  1,  2,  7,  0,  6, 13, 11,  5, 12,\n\t        1,  9, 11, 10,  0,  8, 12,  4, 13,  3,  7, 15, 14,  5,  6,  2,\n\t        4,  0,  5,  9,  7, 12,  2, 10, 14,  1,  3,  8, 11,  6, 15, 13]);\n\t    var _zr = WordArray.create([\n\t        5, 14,  7,  0,  9,  2, 11,  4, 13,  6, 15,  8,  1, 10,  3, 12,\n\t        6, 11,  3,  7,  0, 13,  5, 10, 14, 15,  8, 12,  4,  9,  1,  2,\n\t        15,  5,  1,  3,  7, 14,  6,  9, 11,  8, 12,  2, 10,  0,  4, 13,\n\t        8,  6,  4,  1,  3, 11, 15,  0,  5, 12,  2, 13,  9,  7, 10, 14,\n\t        12, 15, 10,  4,  1,  5,  8,  7,  6,  2, 13, 14,  0,  3,  9, 11]);\n\t    var _sl = WordArray.create([\n\t         11, 14, 15, 12,  5,  8,  7,  9, 11, 13, 14, 15,  6,  7,  9,  8,\n\t        7, 6,   8, 13, 11,  9,  7, 15,  7, 12, 15,  9, 11,  7, 13, 12,\n\t        11, 13,  6,  7, 14,  9, 13, 15, 14,  8, 13,  6,  5, 12,  7,  5,\n\t          11, 12, 14, 15, 14, 15,  9,  8,  9, 14,  5,  6,  8,  6,  5, 12,\n\t        9, 15,  5, 11,  6,  8, 13, 12,  5, 12, 13, 14, 11,  8,  5,  6 ]);\n\t    var _sr = WordArray.create([\n\t        8,  9,  9, 11, 13, 15, 15,  5,  7,  7,  8, 11, 14, 14, 12,  6,\n\t        9, 13, 15,  7, 12,  8,  9, 11,  7,  7, 12,  7,  6, 15, 13, 11,\n\t        9,  7, 15, 11,  8,  6,  6, 14, 12, 13,  5, 14, 13, 13,  7,  5,\n\t        15,  5,  8, 11, 14, 14,  6, 14,  6,  9, 12,  9, 12,  5, 15,  8,\n\t        8,  5, 12,  9, 12,  5, 14,  6,  8, 13,  6,  5, 15, 13, 11, 11 ]);\n\n\t    var _hl =  WordArray.create([ 0x00000000, 0x5A827999, 0x6ED9EBA1, 0x8F1BBCDC, 0xA953FD4E]);\n\t    var _hr =  WordArray.create([ 0x50A28BE6, 0x5C4DD124, 0x6D703EF3, 0x7A6D76E9, 0x00000000]);\n\n\t    /**\n\t     * RIPEMD160 hash algorithm.\n\t     */\n\t    var RIPEMD160 = C_algo.RIPEMD160 = Hasher.extend({\n\t        _doReset: function () {\n\t            this._hash  = WordArray.create([0x67452301, 0xEFCDAB89, 0x98BADCFE, 0x10325476, 0xC3D2E1F0]);\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\n\t            // Swap endian\n\t            for (var i = 0; i < 16; i++) {\n\t                // Shortcuts\n\t                var offset_i = offset + i;\n\t                var M_offset_i = M[offset_i];\n\n\t                // Swap\n\t                M[offset_i] = (\n\t                    (((M_offset_i << 8)  | (M_offset_i >>> 24)) & 0x00ff00ff) |\n\t                    (((M_offset_i << 24) | (M_offset_i >>> 8))  & 0xff00ff00)\n\t                );\n\t            }\n\t            // Shortcut\n\t            var H  = this._hash.words;\n\t            var hl = _hl.words;\n\t            var hr = _hr.words;\n\t            var zl = _zl.words;\n\t            var zr = _zr.words;\n\t            var sl = _sl.words;\n\t            var sr = _sr.words;\n\n\t            // Working variables\n\t            var al, bl, cl, dl, el;\n\t            var ar, br, cr, dr, er;\n\n\t            ar = al = H[0];\n\t            br = bl = H[1];\n\t            cr = cl = H[2];\n\t            dr = dl = H[3];\n\t            er = el = H[4];\n\t            // Computation\n\t            var t;\n\t            for (var i = 0; i < 80; i += 1) {\n\t                t = (al +  M[offset+zl[i]])|0;\n\t                if (i<16){\n\t\t            t +=  f1(bl,cl,dl) + hl[0];\n\t                } else if (i<32) {\n\t\t            t +=  f2(bl,cl,dl) + hl[1];\n\t                } else if (i<48) {\n\t\t            t +=  f3(bl,cl,dl) + hl[2];\n\t                } else if (i<64) {\n\t\t            t +=  f4(bl,cl,dl) + hl[3];\n\t                } else {// if (i<80) {\n\t\t            t +=  f5(bl,cl,dl) + hl[4];\n\t                }\n\t                t = t|0;\n\t                t =  rotl(t,sl[i]);\n\t                t = (t+el)|0;\n\t                al = el;\n\t                el = dl;\n\t                dl = rotl(cl, 10);\n\t                cl = bl;\n\t                bl = t;\n\n\t                t = (ar + M[offset+zr[i]])|0;\n\t                if (i<16){\n\t\t            t +=  f5(br,cr,dr) + hr[0];\n\t                } else if (i<32) {\n\t\t            t +=  f4(br,cr,dr) + hr[1];\n\t                } else if (i<48) {\n\t\t            t +=  f3(br,cr,dr) + hr[2];\n\t                } else if (i<64) {\n\t\t            t +=  f2(br,cr,dr) + hr[3];\n\t                } else {// if (i<80) {\n\t\t            t +=  f1(br,cr,dr) + hr[4];\n\t                }\n\t                t = t|0;\n\t                t =  rotl(t,sr[i]) ;\n\t                t = (t+er)|0;\n\t                ar = er;\n\t                er = dr;\n\t                dr = rotl(cr, 10);\n\t                cr = br;\n\t                br = t;\n\t            }\n\t            // Intermediate hash value\n\t            t    = (H[1] + cl + dr)|0;\n\t            H[1] = (H[2] + dl + er)|0;\n\t            H[2] = (H[3] + el + ar)|0;\n\t            H[3] = (H[4] + al + br)|0;\n\t            H[4] = (H[0] + bl + cr)|0;\n\t            H[0] =  t;\n\t        },\n\n\t        _doFinalize: function () {\n\t            // Shortcuts\n\t            var data = this._data;\n\t            var dataWords = data.words;\n\n\t            var nBitsTotal = this._nDataBytes * 8;\n\t            var nBitsLeft = data.sigBytes * 8;\n\n\t            // Add padding\n\t            dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);\n\t            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = (\n\t                (((nBitsTotal << 8)  | (nBitsTotal >>> 24)) & 0x00ff00ff) |\n\t                (((nBitsTotal << 24) | (nBitsTotal >>> 8))  & 0xff00ff00)\n\t            );\n\t            data.sigBytes = (dataWords.length + 1) * 4;\n\n\t            // Hash final blocks\n\t            this._process();\n\n\t            // Shortcuts\n\t            var hash = this._hash;\n\t            var H = hash.words;\n\n\t            // Swap endian\n\t            for (var i = 0; i < 5; i++) {\n\t                // Shortcut\n\t                var H_i = H[i];\n\n\t                // Swap\n\t                H[i] = (((H_i << 8)  | (H_i >>> 24)) & 0x00ff00ff) |\n\t                       (((H_i << 24) | (H_i >>> 8))  & 0xff00ff00);\n\t            }\n\n\t            // Return final computed hash\n\t            return hash;\n\t        },\n\n\t        clone: function () {\n\t            var clone = Hasher.clone.call(this);\n\t            clone._hash = this._hash.clone();\n\n\t            return clone;\n\t        }\n\t    });\n\n\n\t    function f1(x, y, z) {\n\t        return ((x) ^ (y) ^ (z));\n\n\t    }\n\n\t    function f2(x, y, z) {\n\t        return (((x)&(y)) | ((~x)&(z)));\n\t    }\n\n\t    function f3(x, y, z) {\n\t        return (((x) | (~(y))) ^ (z));\n\t    }\n\n\t    function f4(x, y, z) {\n\t        return (((x) & (z)) | ((y)&(~(z))));\n\t    }\n\n\t    function f5(x, y, z) {\n\t        return ((x) ^ ((y) |(~(z))));\n\n\t    }\n\n\t    function rotl(x,n) {\n\t        return (x<<n) | (x>>>(32-n));\n\t    }\n\n\n\t    /**\n\t     * Shortcut function to the hasher's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     *\n\t     * @return {WordArray} The hash.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hash = CryptoJS.RIPEMD160('message');\n\t     *     var hash = CryptoJS.RIPEMD160(wordArray);\n\t     */\n\t    C.RIPEMD160 = Hasher._createHelper(RIPEMD160);\n\n\t    /**\n\t     * Shortcut function to the HMAC's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     * @param {WordArray|string} key The secret key.\n\t     *\n\t     * @return {WordArray} The HMAC.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hmac = CryptoJS.HmacRIPEMD160(message, key);\n\t     */\n\t    C.HmacRIPEMD160 = Hasher._createHmacHelper(RIPEMD160);\n\t}(Math));\n\n\n\treturn CryptoJS.RIPEMD160;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var Hasher = C_lib.Hasher;\n\t    var C_algo = C.algo;\n\n\t    // Reusable object\n\t    var W = [];\n\n\t    /**\n\t     * SHA-1 hash algorithm.\n\t     */\n\t    var SHA1 = C_algo.SHA1 = Hasher.extend({\n\t        _doReset: function () {\n\t            this._hash = new WordArray.init([\n\t                0x67452301, 0xefcdab89,\n\t                0x98badcfe, 0x10325476,\n\t                0xc3d2e1f0\n\t            ]);\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\t            // Shortcut\n\t            var H = this._hash.words;\n\n\t            // Working variables\n\t            var a = H[0];\n\t            var b = H[1];\n\t            var c = H[2];\n\t            var d = H[3];\n\t            var e = H[4];\n\n\t            // Computation\n\t            for (var i = 0; i < 80; i++) {\n\t                if (i < 16) {\n\t                    W[i] = M[offset + i] | 0;\n\t                } else {\n\t                    var n = W[i - 3] ^ W[i - 8] ^ W[i - 14] ^ W[i - 16];\n\t                    W[i] = (n << 1) | (n >>> 31);\n\t                }\n\n\t                var t = ((a << 5) | (a >>> 27)) + e + W[i];\n\t                if (i < 20) {\n\t                    t += ((b & c) | (~b & d)) + 0x5a827999;\n\t                } else if (i < 40) {\n\t                    t += (b ^ c ^ d) + 0x6ed9eba1;\n\t                } else if (i < 60) {\n\t                    t += ((b & c) | (b & d) | (c & d)) - 0x70e44324;\n\t                } else /* if (i < 80) */ {\n\t                    t += (b ^ c ^ d) - 0x359d3e2a;\n\t                }\n\n\t                e = d;\n\t                d = c;\n\t                c = (b << 30) | (b >>> 2);\n\t                b = a;\n\t                a = t;\n\t            }\n\n\t            // Intermediate hash value\n\t            H[0] = (H[0] + a) | 0;\n\t            H[1] = (H[1] + b) | 0;\n\t            H[2] = (H[2] + c) | 0;\n\t            H[3] = (H[3] + d) | 0;\n\t            H[4] = (H[4] + e) | 0;\n\t        },\n\n\t        _doFinalize: function () {\n\t            // Shortcuts\n\t            var data = this._data;\n\t            var dataWords = data.words;\n\n\t            var nBitsTotal = this._nDataBytes * 8;\n\t            var nBitsLeft = data.sigBytes * 8;\n\n\t            // Add padding\n\t            dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);\n\t            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = Math.floor(nBitsTotal / 0x100000000);\n\t            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 15] = nBitsTotal;\n\t            data.sigBytes = dataWords.length * 4;\n\n\t            // Hash final blocks\n\t            this._process();\n\n\t            // Return final computed hash\n\t            return this._hash;\n\t        },\n\n\t        clone: function () {\n\t            var clone = Hasher.clone.call(this);\n\t            clone._hash = this._hash.clone();\n\n\t            return clone;\n\t        }\n\t    });\n\n\t    /**\n\t     * Shortcut function to the hasher's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     *\n\t     * @return {WordArray} The hash.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hash = CryptoJS.SHA1('message');\n\t     *     var hash = CryptoJS.SHA1(wordArray);\n\t     */\n\t    C.SHA1 = Hasher._createHelper(SHA1);\n\n\t    /**\n\t     * Shortcut function to the HMAC's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     * @param {WordArray|string} key The secret key.\n\t     *\n\t     * @return {WordArray} The HMAC.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hmac = CryptoJS.HmacSHA1(message, key);\n\t     */\n\t    C.HmacSHA1 = Hasher._createHmacHelper(SHA1);\n\t}());\n\n\n\treturn CryptoJS.SHA1;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./sha256\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./sha256\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var C_algo = C.algo;\n\t    var SHA256 = C_algo.SHA256;\n\n\t    /**\n\t     * SHA-224 hash algorithm.\n\t     */\n\t    var SHA224 = C_algo.SHA224 = SHA256.extend({\n\t        _doReset: function () {\n\t            this._hash = new WordArray.init([\n\t                0xc1059ed8, 0x367cd507, 0x3070dd17, 0xf70e5939,\n\t                0xffc00b31, 0x68581511, 0x64f98fa7, 0xbefa4fa4\n\t            ]);\n\t        },\n\n\t        _doFinalize: function () {\n\t            var hash = SHA256._doFinalize.call(this);\n\n\t            hash.sigBytes -= 4;\n\n\t            return hash;\n\t        }\n\t    });\n\n\t    /**\n\t     * Shortcut function to the hasher's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     *\n\t     * @return {WordArray} The hash.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hash = CryptoJS.SHA224('message');\n\t     *     var hash = CryptoJS.SHA224(wordArray);\n\t     */\n\t    C.SHA224 = SHA256._createHelper(SHA224);\n\n\t    /**\n\t     * Shortcut function to the HMAC's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     * @param {WordArray|string} key The secret key.\n\t     *\n\t     * @return {WordArray} The HMAC.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hmac = CryptoJS.HmacSHA224(message, key);\n\t     */\n\t    C.HmacSHA224 = SHA256._createHmacHelper(SHA224);\n\t}());\n\n\n\treturn CryptoJS.SHA224;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function (Math) {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var Hasher = C_lib.Hasher;\n\t    var C_algo = C.algo;\n\n\t    // Initialization and round constants tables\n\t    var H = [];\n\t    var K = [];\n\n\t    // Compute constants\n\t    (function () {\n\t        function isPrime(n) {\n\t            var sqrtN = Math.sqrt(n);\n\t            for (var factor = 2; factor <= sqrtN; factor++) {\n\t                if (!(n % factor)) {\n\t                    return false;\n\t                }\n\t            }\n\n\t            return true;\n\t        }\n\n\t        function getFractionalBits(n) {\n\t            return ((n - (n | 0)) * 0x100000000) | 0;\n\t        }\n\n\t        var n = 2;\n\t        var nPrime = 0;\n\t        while (nPrime < 64) {\n\t            if (isPrime(n)) {\n\t                if (nPrime < 8) {\n\t                    H[nPrime] = getFractionalBits(Math.pow(n, 1 / 2));\n\t                }\n\t                K[nPrime] = getFractionalBits(Math.pow(n, 1 / 3));\n\n\t                nPrime++;\n\t            }\n\n\t            n++;\n\t        }\n\t    }());\n\n\t    // Reusable object\n\t    var W = [];\n\n\t    /**\n\t     * SHA-256 hash algorithm.\n\t     */\n\t    var SHA256 = C_algo.SHA256 = Hasher.extend({\n\t        _doReset: function () {\n\t            this._hash = new WordArray.init(H.slice(0));\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\t            // Shortcut\n\t            var H = this._hash.words;\n\n\t            // Working variables\n\t            var a = H[0];\n\t            var b = H[1];\n\t            var c = H[2];\n\t            var d = H[3];\n\t            var e = H[4];\n\t            var f = H[5];\n\t            var g = H[6];\n\t            var h = H[7];\n\n\t            // Computation\n\t            for (var i = 0; i < 64; i++) {\n\t                if (i < 16) {\n\t                    W[i] = M[offset + i] | 0;\n\t                } else {\n\t                    var gamma0x = W[i - 15];\n\t                    var gamma0  = ((gamma0x << 25) | (gamma0x >>> 7))  ^\n\t                                  ((gamma0x << 14) | (gamma0x >>> 18)) ^\n\t                                   (gamma0x >>> 3);\n\n\t                    var gamma1x = W[i - 2];\n\t                    var gamma1  = ((gamma1x << 15) | (gamma1x >>> 17)) ^\n\t                                  ((gamma1x << 13) | (gamma1x >>> 19)) ^\n\t                                   (gamma1x >>> 10);\n\n\t                    W[i] = gamma0 + W[i - 7] + gamma1 + W[i - 16];\n\t                }\n\n\t                var ch  = (e & f) ^ (~e & g);\n\t                var maj = (a & b) ^ (a & c) ^ (b & c);\n\n\t                var sigma0 = ((a << 30) | (a >>> 2)) ^ ((a << 19) | (a >>> 13)) ^ ((a << 10) | (a >>> 22));\n\t                var sigma1 = ((e << 26) | (e >>> 6)) ^ ((e << 21) | (e >>> 11)) ^ ((e << 7)  | (e >>> 25));\n\n\t                var t1 = h + sigma1 + ch + K[i] + W[i];\n\t                var t2 = sigma0 + maj;\n\n\t                h = g;\n\t                g = f;\n\t                f = e;\n\t                e = (d + t1) | 0;\n\t                d = c;\n\t                c = b;\n\t                b = a;\n\t                a = (t1 + t2) | 0;\n\t            }\n\n\t            // Intermediate hash value\n\t            H[0] = (H[0] + a) | 0;\n\t            H[1] = (H[1] + b) | 0;\n\t            H[2] = (H[2] + c) | 0;\n\t            H[3] = (H[3] + d) | 0;\n\t            H[4] = (H[4] + e) | 0;\n\t            H[5] = (H[5] + f) | 0;\n\t            H[6] = (H[6] + g) | 0;\n\t            H[7] = (H[7] + h) | 0;\n\t        },\n\n\t        _doFinalize: function () {\n\t            // Shortcuts\n\t            var data = this._data;\n\t            var dataWords = data.words;\n\n\t            var nBitsTotal = this._nDataBytes * 8;\n\t            var nBitsLeft = data.sigBytes * 8;\n\n\t            // Add padding\n\t            dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);\n\t            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = Math.floor(nBitsTotal / 0x100000000);\n\t            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 15] = nBitsTotal;\n\t            data.sigBytes = dataWords.length * 4;\n\n\t            // Hash final blocks\n\t            this._process();\n\n\t            // Return final computed hash\n\t            return this._hash;\n\t        },\n\n\t        clone: function () {\n\t            var clone = Hasher.clone.call(this);\n\t            clone._hash = this._hash.clone();\n\n\t            return clone;\n\t        }\n\t    });\n\n\t    /**\n\t     * Shortcut function to the hasher's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     *\n\t     * @return {WordArray} The hash.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hash = CryptoJS.SHA256('message');\n\t     *     var hash = CryptoJS.SHA256(wordArray);\n\t     */\n\t    C.SHA256 = Hasher._createHelper(SHA256);\n\n\t    /**\n\t     * Shortcut function to the HMAC's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     * @param {WordArray|string} key The secret key.\n\t     *\n\t     * @return {WordArray} The HMAC.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hmac = CryptoJS.HmacSHA256(message, key);\n\t     */\n\t    C.HmacSHA256 = Hasher._createHmacHelper(SHA256);\n\t}(Math));\n\n\n\treturn CryptoJS.SHA256;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./x64-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./x64-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function (Math) {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var Hasher = C_lib.Hasher;\n\t    var C_x64 = C.x64;\n\t    var X64Word = C_x64.Word;\n\t    var C_algo = C.algo;\n\n\t    // Constants tables\n\t    var RHO_OFFSETS = [];\n\t    var PI_INDEXES  = [];\n\t    var ROUND_CONSTANTS = [];\n\n\t    // Compute Constants\n\t    (function () {\n\t        // Compute rho offset constants\n\t        var x = 1, y = 0;\n\t        for (var t = 0; t < 24; t++) {\n\t            RHO_OFFSETS[x + 5 * y] = ((t + 1) * (t + 2) / 2) % 64;\n\n\t            var newX = y % 5;\n\t            var newY = (2 * x + 3 * y) % 5;\n\t            x = newX;\n\t            y = newY;\n\t        }\n\n\t        // Compute pi index constants\n\t        for (var x = 0; x < 5; x++) {\n\t            for (var y = 0; y < 5; y++) {\n\t                PI_INDEXES[x + 5 * y] = y + ((2 * x + 3 * y) % 5) * 5;\n\t            }\n\t        }\n\n\t        // Compute round constants\n\t        var LFSR = 0x01;\n\t        for (var i = 0; i < 24; i++) {\n\t            var roundConstantMsw = 0;\n\t            var roundConstantLsw = 0;\n\n\t            for (var j = 0; j < 7; j++) {\n\t                if (LFSR & 0x01) {\n\t                    var bitPosition = (1 << j) - 1;\n\t                    if (bitPosition < 32) {\n\t                        roundConstantLsw ^= 1 << bitPosition;\n\t                    } else /* if (bitPosition >= 32) */ {\n\t                        roundConstantMsw ^= 1 << (bitPosition - 32);\n\t                    }\n\t                }\n\n\t                // Compute next LFSR\n\t                if (LFSR & 0x80) {\n\t                    // Primitive polynomial over GF(2): x^8 + x^6 + x^5 + x^4 + 1\n\t                    LFSR = (LFSR << 1) ^ 0x71;\n\t                } else {\n\t                    LFSR <<= 1;\n\t                }\n\t            }\n\n\t            ROUND_CONSTANTS[i] = X64Word.create(roundConstantMsw, roundConstantLsw);\n\t        }\n\t    }());\n\n\t    // Reusable objects for temporary values\n\t    var T = [];\n\t    (function () {\n\t        for (var i = 0; i < 25; i++) {\n\t            T[i] = X64Word.create();\n\t        }\n\t    }());\n\n\t    /**\n\t     * SHA-3 hash algorithm.\n\t     */\n\t    var SHA3 = C_algo.SHA3 = Hasher.extend({\n\t        /**\n\t         * Configuration options.\n\t         *\n\t         * @property {number} outputLength\n\t         *   The desired number of bits in the output hash.\n\t         *   Only values permitted are: 224, 256, 384, 512.\n\t         *   Default: 512\n\t         */\n\t        cfg: Hasher.cfg.extend({\n\t            outputLength: 512\n\t        }),\n\n\t        _doReset: function () {\n\t            var state = this._state = []\n\t            for (var i = 0; i < 25; i++) {\n\t                state[i] = new X64Word.init();\n\t            }\n\n\t            this.blockSize = (1600 - 2 * this.cfg.outputLength) / 32;\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\t            // Shortcuts\n\t            var state = this._state;\n\t            var nBlockSizeLanes = this.blockSize / 2;\n\n\t            // Absorb\n\t            for (var i = 0; i < nBlockSizeLanes; i++) {\n\t                // Shortcuts\n\t                var M2i  = M[offset + 2 * i];\n\t                var M2i1 = M[offset + 2 * i + 1];\n\n\t                // Swap endian\n\t                M2i = (\n\t                    (((M2i << 8)  | (M2i >>> 24)) & 0x00ff00ff) |\n\t                    (((M2i << 24) | (M2i >>> 8))  & 0xff00ff00)\n\t                );\n\t                M2i1 = (\n\t                    (((M2i1 << 8)  | (M2i1 >>> 24)) & 0x00ff00ff) |\n\t                    (((M2i1 << 24) | (M2i1 >>> 8))  & 0xff00ff00)\n\t                );\n\n\t                // Absorb message into state\n\t                var lane = state[i];\n\t                lane.high ^= M2i1;\n\t                lane.low  ^= M2i;\n\t            }\n\n\t            // Rounds\n\t            for (var round = 0; round < 24; round++) {\n\t                // Theta\n\t                for (var x = 0; x < 5; x++) {\n\t                    // Mix column lanes\n\t                    var tMsw = 0, tLsw = 0;\n\t                    for (var y = 0; y < 5; y++) {\n\t                        var lane = state[x + 5 * y];\n\t                        tMsw ^= lane.high;\n\t                        tLsw ^= lane.low;\n\t                    }\n\n\t                    // Temporary values\n\t                    var Tx = T[x];\n\t                    Tx.high = tMsw;\n\t                    Tx.low  = tLsw;\n\t                }\n\t                for (var x = 0; x < 5; x++) {\n\t                    // Shortcuts\n\t                    var Tx4 = T[(x + 4) % 5];\n\t                    var Tx1 = T[(x + 1) % 5];\n\t                    var Tx1Msw = Tx1.high;\n\t                    var Tx1Lsw = Tx1.low;\n\n\t                    // Mix surrounding columns\n\t                    var tMsw = Tx4.high ^ ((Tx1Msw << 1) | (Tx1Lsw >>> 31));\n\t                    var tLsw = Tx4.low  ^ ((Tx1Lsw << 1) | (Tx1Msw >>> 31));\n\t                    for (var y = 0; y < 5; y++) {\n\t                        var lane = state[x + 5 * y];\n\t                        lane.high ^= tMsw;\n\t                        lane.low  ^= tLsw;\n\t                    }\n\t                }\n\n\t                // Rho Pi\n\t                for (var laneIndex = 1; laneIndex < 25; laneIndex++) {\n\t                    // Shortcuts\n\t                    var lane = state[laneIndex];\n\t                    var laneMsw = lane.high;\n\t                    var laneLsw = lane.low;\n\t                    var rhoOffset = RHO_OFFSETS[laneIndex];\n\n\t                    // Rotate lanes\n\t                    if (rhoOffset < 32) {\n\t                        var tMsw = (laneMsw << rhoOffset) | (laneLsw >>> (32 - rhoOffset));\n\t                        var tLsw = (laneLsw << rhoOffset) | (laneMsw >>> (32 - rhoOffset));\n\t                    } else /* if (rhoOffset >= 32) */ {\n\t                        var tMsw = (laneLsw << (rhoOffset - 32)) | (laneMsw >>> (64 - rhoOffset));\n\t                        var tLsw = (laneMsw << (rhoOffset - 32)) | (laneLsw >>> (64 - rhoOffset));\n\t                    }\n\n\t                    // Transpose lanes\n\t                    var TPiLane = T[PI_INDEXES[laneIndex]];\n\t                    TPiLane.high = tMsw;\n\t                    TPiLane.low  = tLsw;\n\t                }\n\n\t                // Rho pi at x = y = 0\n\t                var T0 = T[0];\n\t                var state0 = state[0];\n\t                T0.high = state0.high;\n\t                T0.low  = state0.low;\n\n\t                // Chi\n\t                for (var x = 0; x < 5; x++) {\n\t                    for (var y = 0; y < 5; y++) {\n\t                        // Shortcuts\n\t                        var laneIndex = x + 5 * y;\n\t                        var lane = state[laneIndex];\n\t                        var TLane = T[laneIndex];\n\t                        var Tx1Lane = T[((x + 1) % 5) + 5 * y];\n\t                        var Tx2Lane = T[((x + 2) % 5) + 5 * y];\n\n\t                        // Mix rows\n\t                        lane.high = TLane.high ^ (~Tx1Lane.high & Tx2Lane.high);\n\t                        lane.low  = TLane.low  ^ (~Tx1Lane.low  & Tx2Lane.low);\n\t                    }\n\t                }\n\n\t                // Iota\n\t                var lane = state[0];\n\t                var roundConstant = ROUND_CONSTANTS[round];\n\t                lane.high ^= roundConstant.high;\n\t                lane.low  ^= roundConstant.low;;\n\t            }\n\t        },\n\n\t        _doFinalize: function () {\n\t            // Shortcuts\n\t            var data = this._data;\n\t            var dataWords = data.words;\n\t            var nBitsTotal = this._nDataBytes * 8;\n\t            var nBitsLeft = data.sigBytes * 8;\n\t            var blockSizeBits = this.blockSize * 32;\n\n\t            // Add padding\n\t            dataWords[nBitsLeft >>> 5] |= 0x1 << (24 - nBitsLeft % 32);\n\t            dataWords[((Math.ceil((nBitsLeft + 1) / blockSizeBits) * blockSizeBits) >>> 5) - 1] |= 0x80;\n\t            data.sigBytes = dataWords.length * 4;\n\n\t            // Hash final blocks\n\t            this._process();\n\n\t            // Shortcuts\n\t            var state = this._state;\n\t            var outputLengthBytes = this.cfg.outputLength / 8;\n\t            var outputLengthLanes = outputLengthBytes / 8;\n\n\t            // Squeeze\n\t            var hashWords = [];\n\t            for (var i = 0; i < outputLengthLanes; i++) {\n\t                // Shortcuts\n\t                var lane = state[i];\n\t                var laneMsw = lane.high;\n\t                var laneLsw = lane.low;\n\n\t                // Swap endian\n\t                laneMsw = (\n\t                    (((laneMsw << 8)  | (laneMsw >>> 24)) & 0x00ff00ff) |\n\t                    (((laneMsw << 24) | (laneMsw >>> 8))  & 0xff00ff00)\n\t                );\n\t                laneLsw = (\n\t                    (((laneLsw << 8)  | (laneLsw >>> 24)) & 0x00ff00ff) |\n\t                    (((laneLsw << 24) | (laneLsw >>> 8))  & 0xff00ff00)\n\t                );\n\n\t                // Squeeze state to retrieve hash\n\t                hashWords.push(laneLsw);\n\t                hashWords.push(laneMsw);\n\t            }\n\n\t            // Return final computed hash\n\t            return new WordArray.init(hashWords, outputLengthBytes);\n\t        },\n\n\t        clone: function () {\n\t            var clone = Hasher.clone.call(this);\n\n\t            var state = clone._state = this._state.slice(0);\n\t            for (var i = 0; i < 25; i++) {\n\t                state[i] = state[i].clone();\n\t            }\n\n\t            return clone;\n\t        }\n\t    });\n\n\t    /**\n\t     * Shortcut function to the hasher's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     *\n\t     * @return {WordArray} The hash.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hash = CryptoJS.SHA3('message');\n\t     *     var hash = CryptoJS.SHA3(wordArray);\n\t     */\n\t    C.SHA3 = Hasher._createHelper(SHA3);\n\n\t    /**\n\t     * Shortcut function to the HMAC's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     * @param {WordArray|string} key The secret key.\n\t     *\n\t     * @return {WordArray} The HMAC.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hmac = CryptoJS.HmacSHA3(message, key);\n\t     */\n\t    C.HmacSHA3 = Hasher._createHmacHelper(SHA3);\n\t}(Math));\n\n\n\treturn CryptoJS.SHA3;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./x64-core\"), require(\"./sha512\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./x64-core\", \"./sha512\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_x64 = C.x64;\n\t    var X64Word = C_x64.Word;\n\t    var X64WordArray = C_x64.WordArray;\n\t    var C_algo = C.algo;\n\t    var SHA512 = C_algo.SHA512;\n\n\t    /**\n\t     * SHA-384 hash algorithm.\n\t     */\n\t    var SHA384 = C_algo.SHA384 = SHA512.extend({\n\t        _doReset: function () {\n\t            this._hash = new X64WordArray.init([\n\t                new X64Word.init(0xcbbb9d5d, 0xc1059ed8), new X64Word.init(0x629a292a, 0x367cd507),\n\t                new X64Word.init(0x9159015a, 0x3070dd17), new X64Word.init(0x152fecd8, 0xf70e5939),\n\t                new X64Word.init(0x67332667, 0xffc00b31), new X64Word.init(0x8eb44a87, 0x68581511),\n\t                new X64Word.init(0xdb0c2e0d, 0x64f98fa7), new X64Word.init(0x47b5481d, 0xbefa4fa4)\n\t            ]);\n\t        },\n\n\t        _doFinalize: function () {\n\t            var hash = SHA512._doFinalize.call(this);\n\n\t            hash.sigBytes -= 16;\n\n\t            return hash;\n\t        }\n\t    });\n\n\t    /**\n\t     * Shortcut function to the hasher's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     *\n\t     * @return {WordArray} The hash.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hash = CryptoJS.SHA384('message');\n\t     *     var hash = CryptoJS.SHA384(wordArray);\n\t     */\n\t    C.SHA384 = SHA512._createHelper(SHA384);\n\n\t    /**\n\t     * Shortcut function to the HMAC's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     * @param {WordArray|string} key The secret key.\n\t     *\n\t     * @return {WordArray} The HMAC.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hmac = CryptoJS.HmacSHA384(message, key);\n\t     */\n\t    C.HmacSHA384 = SHA512._createHmacHelper(SHA384);\n\t}());\n\n\n\treturn CryptoJS.SHA384;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./x64-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./x64-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var Hasher = C_lib.Hasher;\n\t    var C_x64 = C.x64;\n\t    var X64Word = C_x64.Word;\n\t    var X64WordArray = C_x64.WordArray;\n\t    var C_algo = C.algo;\n\n\t    function X64Word_create() {\n\t        return X64Word.create.apply(X64Word, arguments);\n\t    }\n\n\t    // Constants\n\t    var K = [\n\t        X64Word_create(0x428a2f98, 0xd728ae22), X64Word_create(0x71374491, 0x23ef65cd),\n\t        X64Word_create(0xb5c0fbcf, 0xec4d3b2f), X64Word_create(0xe9b5dba5, 0x8189dbbc),\n\t        X64Word_create(0x3956c25b, 0xf348b538), X64Word_create(0x59f111f1, 0xb605d019),\n\t        X64Word_create(0x923f82a4, 0xaf194f9b), X64Word_create(0xab1c5ed5, 0xda6d8118),\n\t        X64Word_create(0xd807aa98, 0xa3030242), X64Word_create(0x12835b01, 0x45706fbe),\n\t        X64Word_create(0x243185be, 0x4ee4b28c), X64Word_create(0x550c7dc3, 0xd5ffb4e2),\n\t        X64Word_create(0x72be5d74, 0xf27b896f), X64Word_create(0x80deb1fe, 0x3b1696b1),\n\t        X64Word_create(0x9bdc06a7, 0x25c71235), X64Word_create(0xc19bf174, 0xcf692694),\n\t        X64Word_create(0xe49b69c1, 0x9ef14ad2), X64Word_create(0xefbe4786, 0x384f25e3),\n\t        X64Word_create(0x0fc19dc6, 0x8b8cd5b5), X64Word_create(0x240ca1cc, 0x77ac9c65),\n\t        X64Word_create(0x2de92c6f, 0x592b0275), X64Word_create(0x4a7484aa, 0x6ea6e483),\n\t        X64Word_create(0x5cb0a9dc, 0xbd41fbd4), X64Word_create(0x76f988da, 0x831153b5),\n\t        X64Word_create(0x983e5152, 0xee66dfab), X64Word_create(0xa831c66d, 0x2db43210),\n\t        X64Word_create(0xb00327c8, 0x98fb213f), X64Word_create(0xbf597fc7, 0xbeef0ee4),\n\t        X64Word_create(0xc6e00bf3, 0x3da88fc2), X64Word_create(0xd5a79147, 0x930aa725),\n\t        X64Word_create(0x06ca6351, 0xe003826f), X64Word_create(0x14292967, 0x0a0e6e70),\n\t        X64Word_create(0x27b70a85, 0x46d22ffc), X64Word_create(0x2e1b2138, 0x5c26c926),\n\t        X64Word_create(0x4d2c6dfc, 0x5ac42aed), X64Word_create(0x53380d13, 0x9d95b3df),\n\t        X64Word_create(0x650a7354, 0x8baf63de), X64Word_create(0x766a0abb, 0x3c77b2a8),\n\t        X64Word_create(0x81c2c92e, 0x47edaee6), X64Word_create(0x92722c85, 0x1482353b),\n\t        X64Word_create(0xa2bfe8a1, 0x4cf10364), X64Word_create(0xa81a664b, 0xbc423001),\n\t        X64Word_create(0xc24b8b70, 0xd0f89791), X64Word_create(0xc76c51a3, 0x0654be30),\n\t        X64Word_create(0xd192e819, 0xd6ef5218), X64Word_create(0xd6990624, 0x5565a910),\n\t        X64Word_create(0xf40e3585, 0x5771202a), X64Word_create(0x106aa070, 0x32bbd1b8),\n\t        X64Word_create(0x19a4c116, 0xb8d2d0c8), X64Word_create(0x1e376c08, 0x5141ab53),\n\t        X64Word_create(0x2748774c, 0xdf8eeb99), X64Word_create(0x34b0bcb5, 0xe19b48a8),\n\t        X64Word_create(0x391c0cb3, 0xc5c95a63), X64Word_create(0x4ed8aa4a, 0xe3418acb),\n\t        X64Word_create(0x5b9cca4f, 0x7763e373), X64Word_create(0x682e6ff3, 0xd6b2b8a3),\n\t        X64Word_create(0x748f82ee, 0x5defb2fc), X64Word_create(0x78a5636f, 0x43172f60),\n\t        X64Word_create(0x84c87814, 0xa1f0ab72), X64Word_create(0x8cc70208, 0x1a6439ec),\n\t        X64Word_create(0x90befffa, 0x23631e28), X64Word_create(0xa4506ceb, 0xde82bde9),\n\t        X64Word_create(0xbef9a3f7, 0xb2c67915), X64Word_create(0xc67178f2, 0xe372532b),\n\t        X64Word_create(0xca273ece, 0xea26619c), X64Word_create(0xd186b8c7, 0x21c0c207),\n\t        X64Word_create(0xeada7dd6, 0xcde0eb1e), X64Word_create(0xf57d4f7f, 0xee6ed178),\n\t        X64Word_create(0x06f067aa, 0x72176fba), X64Word_create(0x0a637dc5, 0xa2c898a6),\n\t        X64Word_create(0x113f9804, 0xbef90dae), X64Word_create(0x1b710b35, 0x131c471b),\n\t        X64Word_create(0x28db77f5, 0x23047d84), X64Word_create(0x32caab7b, 0x40c72493),\n\t        X64Word_create(0x3c9ebe0a, 0x15c9bebc), X64Word_create(0x431d67c4, 0x9c100d4c),\n\t        X64Word_create(0x4cc5d4be, 0xcb3e42b6), X64Word_create(0x597f299c, 0xfc657e2a),\n\t        X64Word_create(0x5fcb6fab, 0x3ad6faec), X64Word_create(0x6c44198c, 0x4a475817)\n\t    ];\n\n\t    // Reusable objects\n\t    var W = [];\n\t    (function () {\n\t        for (var i = 0; i < 80; i++) {\n\t            W[i] = X64Word_create();\n\t        }\n\t    }());\n\n\t    /**\n\t     * SHA-512 hash algorithm.\n\t     */\n\t    var SHA512 = C_algo.SHA512 = Hasher.extend({\n\t        _doReset: function () {\n\t            this._hash = new X64WordArray.init([\n\t                new X64Word.init(0x6a09e667, 0xf3bcc908), new X64Word.init(0xbb67ae85, 0x84caa73b),\n\t                new X64Word.init(0x3c6ef372, 0xfe94f82b), new X64Word.init(0xa54ff53a, 0x5f1d36f1),\n\t                new X64Word.init(0x510e527f, 0xade682d1), new X64Word.init(0x9b05688c, 0x2b3e6c1f),\n\t                new X64Word.init(0x1f83d9ab, 0xfb41bd6b), new X64Word.init(0x5be0cd19, 0x137e2179)\n\t            ]);\n\t        },\n\n\t        _doProcessBlock: function (M, offset) {\n\t            // Shortcuts\n\t            var H = this._hash.words;\n\n\t            var H0 = H[0];\n\t            var H1 = H[1];\n\t            var H2 = H[2];\n\t            var H3 = H[3];\n\t            var H4 = H[4];\n\t            var H5 = H[5];\n\t            var H6 = H[6];\n\t            var H7 = H[7];\n\n\t            var H0h = H0.high;\n\t            var H0l = H0.low;\n\t            var H1h = H1.high;\n\t            var H1l = H1.low;\n\t            var H2h = H2.high;\n\t            var H2l = H2.low;\n\t            var H3h = H3.high;\n\t            var H3l = H3.low;\n\t            var H4h = H4.high;\n\t            var H4l = H4.low;\n\t            var H5h = H5.high;\n\t            var H5l = H5.low;\n\t            var H6h = H6.high;\n\t            var H6l = H6.low;\n\t            var H7h = H7.high;\n\t            var H7l = H7.low;\n\n\t            // Working variables\n\t            var ah = H0h;\n\t            var al = H0l;\n\t            var bh = H1h;\n\t            var bl = H1l;\n\t            var ch = H2h;\n\t            var cl = H2l;\n\t            var dh = H3h;\n\t            var dl = H3l;\n\t            var eh = H4h;\n\t            var el = H4l;\n\t            var fh = H5h;\n\t            var fl = H5l;\n\t            var gh = H6h;\n\t            var gl = H6l;\n\t            var hh = H7h;\n\t            var hl = H7l;\n\n\t            // Rounds\n\t            for (var i = 0; i < 80; i++) {\n\t                // Shortcut\n\t                var Wi = W[i];\n\n\t                // Extend message\n\t                if (i < 16) {\n\t                    var Wih = Wi.high = M[offset + i * 2]     | 0;\n\t                    var Wil = Wi.low  = M[offset + i * 2 + 1] | 0;\n\t                } else {\n\t                    // Gamma0\n\t                    var gamma0x  = W[i - 15];\n\t                    var gamma0xh = gamma0x.high;\n\t                    var gamma0xl = gamma0x.low;\n\t                    var gamma0h  = ((gamma0xh >>> 1) | (gamma0xl << 31)) ^ ((gamma0xh >>> 8) | (gamma0xl << 24)) ^ (gamma0xh >>> 7);\n\t                    var gamma0l  = ((gamma0xl >>> 1) | (gamma0xh << 31)) ^ ((gamma0xl >>> 8) | (gamma0xh << 24)) ^ ((gamma0xl >>> 7) | (gamma0xh << 25));\n\n\t                    // Gamma1\n\t                    var gamma1x  = W[i - 2];\n\t                    var gamma1xh = gamma1x.high;\n\t                    var gamma1xl = gamma1x.low;\n\t                    var gamma1h  = ((gamma1xh >>> 19) | (gamma1xl << 13)) ^ ((gamma1xh << 3) | (gamma1xl >>> 29)) ^ (gamma1xh >>> 6);\n\t                    var gamma1l  = ((gamma1xl >>> 19) | (gamma1xh << 13)) ^ ((gamma1xl << 3) | (gamma1xh >>> 29)) ^ ((gamma1xl >>> 6) | (gamma1xh << 26));\n\n\t                    // W[i] = gamma0 + W[i - 7] + gamma1 + W[i - 16]\n\t                    var Wi7  = W[i - 7];\n\t                    var Wi7h = Wi7.high;\n\t                    var Wi7l = Wi7.low;\n\n\t                    var Wi16  = W[i - 16];\n\t                    var Wi16h = Wi16.high;\n\t                    var Wi16l = Wi16.low;\n\n\t                    var Wil = gamma0l + Wi7l;\n\t                    var Wih = gamma0h + Wi7h + ((Wil >>> 0) < (gamma0l >>> 0) ? 1 : 0);\n\t                    var Wil = Wil + gamma1l;\n\t                    var Wih = Wih + gamma1h + ((Wil >>> 0) < (gamma1l >>> 0) ? 1 : 0);\n\t                    var Wil = Wil + Wi16l;\n\t                    var Wih = Wih + Wi16h + ((Wil >>> 0) < (Wi16l >>> 0) ? 1 : 0);\n\n\t                    Wi.high = Wih;\n\t                    Wi.low  = Wil;\n\t                }\n\n\t                var chh  = (eh & fh) ^ (~eh & gh);\n\t                var chl  = (el & fl) ^ (~el & gl);\n\t                var majh = (ah & bh) ^ (ah & ch) ^ (bh & ch);\n\t                var majl = (al & bl) ^ (al & cl) ^ (bl & cl);\n\n\t                var sigma0h = ((ah >>> 28) | (al << 4))  ^ ((ah << 30)  | (al >>> 2)) ^ ((ah << 25) | (al >>> 7));\n\t                var sigma0l = ((al >>> 28) | (ah << 4))  ^ ((al << 30)  | (ah >>> 2)) ^ ((al << 25) | (ah >>> 7));\n\t                var sigma1h = ((eh >>> 14) | (el << 18)) ^ ((eh >>> 18) | (el << 14)) ^ ((eh << 23) | (el >>> 9));\n\t                var sigma1l = ((el >>> 14) | (eh << 18)) ^ ((el >>> 18) | (eh << 14)) ^ ((el << 23) | (eh >>> 9));\n\n\t                // t1 = h + sigma1 + ch + K[i] + W[i]\n\t                var Ki  = K[i];\n\t                var Kih = Ki.high;\n\t                var Kil = Ki.low;\n\n\t                var t1l = hl + sigma1l;\n\t                var t1h = hh + sigma1h + ((t1l >>> 0) < (hl >>> 0) ? 1 : 0);\n\t                var t1l = t1l + chl;\n\t                var t1h = t1h + chh + ((t1l >>> 0) < (chl >>> 0) ? 1 : 0);\n\t                var t1l = t1l + Kil;\n\t                var t1h = t1h + Kih + ((t1l >>> 0) < (Kil >>> 0) ? 1 : 0);\n\t                var t1l = t1l + Wil;\n\t                var t1h = t1h + Wih + ((t1l >>> 0) < (Wil >>> 0) ? 1 : 0);\n\n\t                // t2 = sigma0 + maj\n\t                var t2l = sigma0l + majl;\n\t                var t2h = sigma0h + majh + ((t2l >>> 0) < (sigma0l >>> 0) ? 1 : 0);\n\n\t                // Update working variables\n\t                hh = gh;\n\t                hl = gl;\n\t                gh = fh;\n\t                gl = fl;\n\t                fh = eh;\n\t                fl = el;\n\t                el = (dl + t1l) | 0;\n\t                eh = (dh + t1h + ((el >>> 0) < (dl >>> 0) ? 1 : 0)) | 0;\n\t                dh = ch;\n\t                dl = cl;\n\t                ch = bh;\n\t                cl = bl;\n\t                bh = ah;\n\t                bl = al;\n\t                al = (t1l + t2l) | 0;\n\t                ah = (t1h + t2h + ((al >>> 0) < (t1l >>> 0) ? 1 : 0)) | 0;\n\t            }\n\n\t            // Intermediate hash value\n\t            H0l = H0.low  = (H0l + al);\n\t            H0.high = (H0h + ah + ((H0l >>> 0) < (al >>> 0) ? 1 : 0));\n\t            H1l = H1.low  = (H1l + bl);\n\t            H1.high = (H1h + bh + ((H1l >>> 0) < (bl >>> 0) ? 1 : 0));\n\t            H2l = H2.low  = (H2l + cl);\n\t            H2.high = (H2h + ch + ((H2l >>> 0) < (cl >>> 0) ? 1 : 0));\n\t            H3l = H3.low  = (H3l + dl);\n\t            H3.high = (H3h + dh + ((H3l >>> 0) < (dl >>> 0) ? 1 : 0));\n\t            H4l = H4.low  = (H4l + el);\n\t            H4.high = (H4h + eh + ((H4l >>> 0) < (el >>> 0) ? 1 : 0));\n\t            H5l = H5.low  = (H5l + fl);\n\t            H5.high = (H5h + fh + ((H5l >>> 0) < (fl >>> 0) ? 1 : 0));\n\t            H6l = H6.low  = (H6l + gl);\n\t            H6.high = (H6h + gh + ((H6l >>> 0) < (gl >>> 0) ? 1 : 0));\n\t            H7l = H7.low  = (H7l + hl);\n\t            H7.high = (H7h + hh + ((H7l >>> 0) < (hl >>> 0) ? 1 : 0));\n\t        },\n\n\t        _doFinalize: function () {\n\t            // Shortcuts\n\t            var data = this._data;\n\t            var dataWords = data.words;\n\n\t            var nBitsTotal = this._nDataBytes * 8;\n\t            var nBitsLeft = data.sigBytes * 8;\n\n\t            // Add padding\n\t            dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);\n\t            dataWords[(((nBitsLeft + 128) >>> 10) << 5) + 30] = Math.floor(nBitsTotal / 0x100000000);\n\t            dataWords[(((nBitsLeft + 128) >>> 10) << 5) + 31] = nBitsTotal;\n\t            data.sigBytes = dataWords.length * 4;\n\n\t            // Hash final blocks\n\t            this._process();\n\n\t            // Convert hash to 32-bit word array before returning\n\t            var hash = this._hash.toX32();\n\n\t            // Return final computed hash\n\t            return hash;\n\t        },\n\n\t        clone: function () {\n\t            var clone = Hasher.clone.call(this);\n\t            clone._hash = this._hash.clone();\n\n\t            return clone;\n\t        },\n\n\t        blockSize: 1024/32\n\t    });\n\n\t    /**\n\t     * Shortcut function to the hasher's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     *\n\t     * @return {WordArray} The hash.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hash = CryptoJS.SHA512('message');\n\t     *     var hash = CryptoJS.SHA512(wordArray);\n\t     */\n\t    C.SHA512 = Hasher._createHelper(SHA512);\n\n\t    /**\n\t     * Shortcut function to the HMAC's object interface.\n\t     *\n\t     * @param {WordArray|string} message The message to hash.\n\t     * @param {WordArray|string} key The secret key.\n\t     *\n\t     * @return {WordArray} The HMAC.\n\t     *\n\t     * @static\n\t     *\n\t     * @example\n\t     *\n\t     *     var hmac = CryptoJS.HmacSHA512(message, key);\n\t     */\n\t    C.HmacSHA512 = Hasher._createHmacHelper(SHA512);\n\t}());\n\n\n\treturn CryptoJS.SHA512;\n\n}));",";(function (root, factory, undef) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"), require(\"./enc-base64\"), require(\"./md5\"), require(\"./evpkdf\"), require(\"./cipher-core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\", \"./enc-base64\", \"./md5\", \"./evpkdf\", \"./cipher-core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var WordArray = C_lib.WordArray;\n\t    var BlockCipher = C_lib.BlockCipher;\n\t    var C_algo = C.algo;\n\n\t    // Permuted Choice 1 constants\n\t    var PC1 = [\n\t        57, 49, 41, 33, 25, 17, 9,  1,\n\t        58, 50, 42, 34, 26, 18, 10, 2,\n\t        59, 51, 43, 35, 27, 19, 11, 3,\n\t        60, 52, 44, 36, 63, 55, 47, 39,\n\t        31, 23, 15, 7,  62, 54, 46, 38,\n\t        30, 22, 14, 6,  61, 53, 45, 37,\n\t        29, 21, 13, 5,  28, 20, 12, 4\n\t    ];\n\n\t    // Permuted Choice 2 constants\n\t    var PC2 = [\n\t        14, 17, 11, 24, 1,  5,\n\t        3,  28, 15, 6,  21, 10,\n\t        23, 19, 12, 4,  26, 8,\n\t        16, 7,  27, 20, 13, 2,\n\t        41, 52, 31, 37, 47, 55,\n\t        30, 40, 51, 45, 33, 48,\n\t        44, 49, 39, 56, 34, 53,\n\t        46, 42, 50, 36, 29, 32\n\t    ];\n\n\t    // Cumulative bit shift constants\n\t    var BIT_SHIFTS = [1,  2,  4,  6,  8,  10, 12, 14, 15, 17, 19, 21, 23, 25, 27, 28];\n\n\t    // SBOXes and round permutation constants\n\t    var SBOX_P = [\n\t        {\n\t            0x0: 0x808200,\n\t            0x10000000: 0x8000,\n\t            0x20000000: 0x808002,\n\t            0x30000000: 0x2,\n\t            0x40000000: 0x200,\n\t            0x50000000: 0x808202,\n\t            0x60000000: 0x800202,\n\t            0x70000000: 0x800000,\n\t            0x80000000: 0x202,\n\t            0x90000000: 0x800200,\n\t            0xa0000000: 0x8200,\n\t            0xb0000000: 0x808000,\n\t            0xc0000000: 0x8002,\n\t            0xd0000000: 0x800002,\n\t            0xe0000000: 0x0,\n\t            0xf0000000: 0x8202,\n\t            0x8000000: 0x0,\n\t            0x18000000: 0x808202,\n\t            0x28000000: 0x8202,\n\t            0x38000000: 0x8000,\n\t            0x48000000: 0x808200,\n\t            0x58000000: 0x200,\n\t            0x68000000: 0x808002,\n\t            0x78000000: 0x2,\n\t            0x88000000: 0x800200,\n\t            0x98000000: 0x8200,\n\t            0xa8000000: 0x808000,\n\t            0xb8000000: 0x800202,\n\t            0xc8000000: 0x800002,\n\t            0xd8000000: 0x8002,\n\t            0xe8000000: 0x202,\n\t            0xf8000000: 0x800000,\n\t            0x1: 0x8000,\n\t            0x10000001: 0x2,\n\t            0x20000001: 0x808200,\n\t            0x30000001: 0x800000,\n\t            0x40000001: 0x808002,\n\t            0x50000001: 0x8200,\n\t            0x60000001: 0x200,\n\t            0x70000001: 0x800202,\n\t            0x80000001: 0x808202,\n\t            0x90000001: 0x808000,\n\t            0xa0000001: 0x800002,\n\t            0xb0000001: 0x8202,\n\t            0xc0000001: 0x202,\n\t            0xd0000001: 0x800200,\n\t            0xe0000001: 0x8002,\n\t            0xf0000001: 0x0,\n\t            0x8000001: 0x808202,\n\t            0x18000001: 0x808000,\n\t            0x28000001: 0x800000,\n\t            0x38000001: 0x200,\n\t            0x48000001: 0x8000,\n\t            0x58000001: 0x800002,\n\t            0x68000001: 0x2,\n\t            0x78000001: 0x8202,\n\t            0x88000001: 0x8002,\n\t            0x98000001: 0x800202,\n\t            0xa8000001: 0x202,\n\t            0xb8000001: 0x808200,\n\t            0xc8000001: 0x800200,\n\t            0xd8000001: 0x0,\n\t            0xe8000001: 0x8200,\n\t            0xf8000001: 0x808002\n\t        },\n\t        {\n\t            0x0: 0x40084010,\n\t            0x1000000: 0x4000,\n\t            0x2000000: 0x80000,\n\t            0x3000000: 0x40080010,\n\t            0x4000000: 0x40000010,\n\t            0x5000000: 0x40084000,\n\t            0x6000000: 0x40004000,\n\t            0x7000000: 0x10,\n\t            0x8000000: 0x84000,\n\t            0x9000000: 0x40004010,\n\t            0xa000000: 0x40000000,\n\t            0xb000000: 0x84010,\n\t            0xc000000: 0x80010,\n\t            0xd000000: 0x0,\n\t            0xe000000: 0x4010,\n\t            0xf000000: 0x40080000,\n\t            0x800000: 0x40004000,\n\t            0x1800000: 0x84010,\n\t            0x2800000: 0x10,\n\t            0x3800000: 0x40004010,\n\t            0x4800000: 0x40084010,\n\t            0x5800000: 0x40000000,\n\t            0x6800000: 0x80000,\n\t            0x7800000: 0x40080010,\n\t            0x8800000: 0x80010,\n\t            0x9800000: 0x0,\n\t            0xa800000: 0x4000,\n\t            0xb800000: 0x40080000,\n\t            0xc800000: 0x40000010,\n\t            0xd800000: 0x84000,\n\t            0xe800000: 0x40084000,\n\t            0xf800000: 0x4010,\n\t            0x10000000: 0x0,\n\t            0x11000000: 0x40080010,\n\t            0x12000000: 0x40004010,\n\t            0x13000000: 0x40084000,\n\t            0x14000000: 0x40080000,\n\t            0x15000000: 0x10,\n\t            0x16000000: 0x84010,\n\t            0x17000000: 0x4000,\n\t            0x18000000: 0x4010,\n\t            0x19000000: 0x80000,\n\t            0x1a000000: 0x80010,\n\t            0x1b000000: 0x40000010,\n\t            0x1c000000: 0x84000,\n\t            0x1d000000: 0x40004000,\n\t            0x1e000000: 0x40000000,\n\t            0x1f000000: 0x40084010,\n\t            0x10800000: 0x84010,\n\t            0x11800000: 0x80000,\n\t            0x12800000: 0x40080000,\n\t            0x13800000: 0x4000,\n\t            0x14800000: 0x40004000,\n\t            0x15800000: 0x40084010,\n\t            0x16800000: 0x10,\n\t            0x17800000: 0x40000000,\n\t            0x18800000: 0x40084000,\n\t            0x19800000: 0x40000010,\n\t            0x1a800000: 0x40004010,\n\t            0x1b800000: 0x80010,\n\t            0x1c800000: 0x0,\n\t            0x1d800000: 0x4010,\n\t            0x1e800000: 0x40080010,\n\t            0x1f800000: 0x84000\n\t        },\n\t        {\n\t            0x0: 0x104,\n\t            0x100000: 0x0,\n\t            0x200000: 0x4000100,\n\t            0x300000: 0x10104,\n\t            0x400000: 0x10004,\n\t            0x500000: 0x4000004,\n\t            0x600000: 0x4010104,\n\t            0x700000: 0x4010000,\n\t            0x800000: 0x4000000,\n\t            0x900000: 0x4010100,\n\t            0xa00000: 0x10100,\n\t            0xb00000: 0x4010004,\n\t            0xc00000: 0x4000104,\n\t            0xd00000: 0x10000,\n\t            0xe00000: 0x4,\n\t            0xf00000: 0x100,\n\t            0x80000: 0x4010100,\n\t            0x180000: 0x4010004,\n\t            0x280000: 0x0,\n\t            0x380000: 0x4000100,\n\t            0x480000: 0x4000004,\n\t            0x580000: 0x10000,\n\t            0x680000: 0x10004,\n\t            0x780000: 0x104,\n\t            0x880000: 0x4,\n\t            0x980000: 0x100,\n\t            0xa80000: 0x4010000,\n\t            0xb80000: 0x10104,\n\t            0xc80000: 0x10100,\n\t            0xd80000: 0x4000104,\n\t            0xe80000: 0x4010104,\n\t            0xf80000: 0x4000000,\n\t            0x1000000: 0x4010100,\n\t            0x1100000: 0x10004,\n\t            0x1200000: 0x10000,\n\t            0x1300000: 0x4000100,\n\t            0x1400000: 0x100,\n\t            0x1500000: 0x4010104,\n\t            0x1600000: 0x4000004,\n\t            0x1700000: 0x0,\n\t            0x1800000: 0x4000104,\n\t            0x1900000: 0x4000000,\n\t            0x1a00000: 0x4,\n\t            0x1b00000: 0x10100,\n\t            0x1c00000: 0x4010000,\n\t            0x1d00000: 0x104,\n\t            0x1e00000: 0x10104,\n\t            0x1f00000: 0x4010004,\n\t            0x1080000: 0x4000000,\n\t            0x1180000: 0x104,\n\t            0x1280000: 0x4010100,\n\t            0x1380000: 0x0,\n\t            0x1480000: 0x10004,\n\t            0x1580000: 0x4000100,\n\t            0x1680000: 0x100,\n\t            0x1780000: 0x4010004,\n\t            0x1880000: 0x10000,\n\t            0x1980000: 0x4010104,\n\t            0x1a80000: 0x10104,\n\t            0x1b80000: 0x4000004,\n\t            0x1c80000: 0x4000104,\n\t            0x1d80000: 0x4010000,\n\t            0x1e80000: 0x4,\n\t            0x1f80000: 0x10100\n\t        },\n\t        {\n\t            0x0: 0x80401000,\n\t            0x10000: 0x80001040,\n\t            0x20000: 0x401040,\n\t            0x30000: 0x80400000,\n\t            0x40000: 0x0,\n\t            0x50000: 0x401000,\n\t            0x60000: 0x80000040,\n\t            0x70000: 0x400040,\n\t            0x80000: 0x80000000,\n\t            0x90000: 0x400000,\n\t            0xa0000: 0x40,\n\t            0xb0000: 0x80001000,\n\t            0xc0000: 0x80400040,\n\t            0xd0000: 0x1040,\n\t            0xe0000: 0x1000,\n\t            0xf0000: 0x80401040,\n\t            0x8000: 0x80001040,\n\t            0x18000: 0x40,\n\t            0x28000: 0x80400040,\n\t            0x38000: 0x80001000,\n\t            0x48000: 0x401000,\n\t            0x58000: 0x80401040,\n\t            0x68000: 0x0,\n\t            0x78000: 0x80400000,\n\t            0x88000: 0x1000,\n\t            0x98000: 0x80401000,\n\t            0xa8000: 0x400000,\n\t            0xb8000: 0x1040,\n\t            0xc8000: 0x80000000,\n\t            0xd8000: 0x400040,\n\t            0xe8000: 0x401040,\n\t            0xf8000: 0x80000040,\n\t            0x100000: 0x400040,\n\t            0x110000: 0x401000,\n\t            0x120000: 0x80000040,\n\t            0x130000: 0x0,\n\t            0x140000: 0x1040,\n\t            0x150000: 0x80400040,\n\t            0x160000: 0x80401000,\n\t            0x170000: 0x80001040,\n\t            0x180000: 0x80401040,\n\t            0x190000: 0x80000000,\n\t            0x1a0000: 0x80400000,\n\t            0x1b0000: 0x401040,\n\t            0x1c0000: 0x80001000,\n\t            0x1d0000: 0x400000,\n\t            0x1e0000: 0x40,\n\t            0x1f0000: 0x1000,\n\t            0x108000: 0x80400000,\n\t            0x118000: 0x80401040,\n\t            0x128000: 0x0,\n\t            0x138000: 0x401000,\n\t            0x148000: 0x400040,\n\t            0x158000: 0x80000000,\n\t            0x168000: 0x80001040,\n\t            0x178000: 0x40,\n\t            0x188000: 0x80000040,\n\t            0x198000: 0x1000,\n\t            0x1a8000: 0x80001000,\n\t            0x1b8000: 0x80400040,\n\t            0x1c8000: 0x1040,\n\t            0x1d8000: 0x80401000,\n\t            0x1e8000: 0x400000,\n\t            0x1f8000: 0x401040\n\t        },\n\t        {\n\t            0x0: 0x80,\n\t            0x1000: 0x1040000,\n\t            0x2000: 0x40000,\n\t            0x3000: 0x20000000,\n\t            0x4000: 0x20040080,\n\t            0x5000: 0x1000080,\n\t            0x6000: 0x21000080,\n\t            0x7000: 0x40080,\n\t            0x8000: 0x1000000,\n\t            0x9000: 0x20040000,\n\t            0xa000: 0x20000080,\n\t            0xb000: 0x21040080,\n\t            0xc000: 0x21040000,\n\t            0xd000: 0x0,\n\t            0xe000: 0x1040080,\n\t            0xf000: 0x21000000,\n\t            0x800: 0x1040080,\n\t            0x1800: 0x21000080,\n\t            0x2800: 0x80,\n\t            0x3800: 0x1040000,\n\t            0x4800: 0x40000,\n\t            0x5800: 0x20040080,\n\t            0x6800: 0x21040000,\n\t            0x7800: 0x20000000,\n\t            0x8800: 0x20040000,\n\t            0x9800: 0x0,\n\t            0xa800: 0x21040080,\n\t            0xb800: 0x1000080,\n\t            0xc800: 0x20000080,\n\t            0xd800: 0x21000000,\n\t            0xe800: 0x1000000,\n\t            0xf800: 0x40080,\n\t            0x10000: 0x40000,\n\t            0x11000: 0x80,\n\t            0x12000: 0x20000000,\n\t            0x13000: 0x21000080,\n\t            0x14000: 0x1000080,\n\t            0x15000: 0x21040000,\n\t            0x16000: 0x20040080,\n\t            0x17000: 0x1000000,\n\t            0x18000: 0x21040080,\n\t            0x19000: 0x21000000,\n\t            0x1a000: 0x1040000,\n\t            0x1b000: 0x20040000,\n\t            0x1c000: 0x40080,\n\t            0x1d000: 0x20000080,\n\t            0x1e000: 0x0,\n\t            0x1f000: 0x1040080,\n\t            0x10800: 0x21000080,\n\t            0x11800: 0x1000000,\n\t            0x12800: 0x1040000,\n\t            0x13800: 0x20040080,\n\t            0x14800: 0x20000000,\n\t            0x15800: 0x1040080,\n\t            0x16800: 0x80,\n\t            0x17800: 0x21040000,\n\t            0x18800: 0x40080,\n\t            0x19800: 0x21040080,\n\t            0x1a800: 0x0,\n\t            0x1b800: 0x21000000,\n\t            0x1c800: 0x1000080,\n\t            0x1d800: 0x40000,\n\t            0x1e800: 0x20040000,\n\t            0x1f800: 0x20000080\n\t        },\n\t        {\n\t            0x0: 0x10000008,\n\t            0x100: 0x2000,\n\t            0x200: 0x10200000,\n\t            0x300: 0x10202008,\n\t            0x400: 0x10002000,\n\t            0x500: 0x200000,\n\t            0x600: 0x200008,\n\t            0x700: 0x10000000,\n\t            0x800: 0x0,\n\t            0x900: 0x10002008,\n\t            0xa00: 0x202000,\n\t            0xb00: 0x8,\n\t            0xc00: 0x10200008,\n\t            0xd00: 0x202008,\n\t            0xe00: 0x2008,\n\t            0xf00: 0x10202000,\n\t            0x80: 0x10200000,\n\t            0x180: 0x10202008,\n\t            0x280: 0x8,\n\t            0x380: 0x200000,\n\t            0x480: 0x202008,\n\t            0x580: 0x10000008,\n\t            0x680: 0x10002000,\n\t            0x780: 0x2008,\n\t            0x880: 0x200008,\n\t            0x980: 0x2000,\n\t            0xa80: 0x10002008,\n\t            0xb80: 0x10200008,\n\t            0xc80: 0x0,\n\t            0xd80: 0x10202000,\n\t            0xe80: 0x202000,\n\t            0xf80: 0x10000000,\n\t            0x1000: 0x10002000,\n\t            0x1100: 0x10200008,\n\t            0x1200: 0x10202008,\n\t            0x1300: 0x2008,\n\t            0x1400: 0x200000,\n\t            0x1500: 0x10000000,\n\t            0x1600: 0x10000008,\n\t            0x1700: 0x202000,\n\t            0x1800: 0x202008,\n\t            0x1900: 0x0,\n\t            0x1a00: 0x8,\n\t            0x1b00: 0x10200000,\n\t            0x1c00: 0x2000,\n\t            0x1d00: 0x10002008,\n\t            0x1e00: 0x10202000,\n\t            0x1f00: 0x200008,\n\t            0x1080: 0x8,\n\t            0x1180: 0x202000,\n\t            0x1280: 0x200000,\n\t            0x1380: 0x10000008,\n\t            0x1480: 0x10002000,\n\t            0x1580: 0x2008,\n\t            0x1680: 0x10202008,\n\t            0x1780: 0x10200000,\n\t            0x1880: 0x10202000,\n\t            0x1980: 0x10200008,\n\t            0x1a80: 0x2000,\n\t            0x1b80: 0x202008,\n\t            0x1c80: 0x200008,\n\t            0x1d80: 0x0,\n\t            0x1e80: 0x10000000,\n\t            0x1f80: 0x10002008\n\t        },\n\t        {\n\t            0x0: 0x100000,\n\t            0x10: 0x2000401,\n\t            0x20: 0x400,\n\t            0x30: 0x100401,\n\t            0x40: 0x2100401,\n\t            0x50: 0x0,\n\t            0x60: 0x1,\n\t            0x70: 0x2100001,\n\t            0x80: 0x2000400,\n\t            0x90: 0x100001,\n\t            0xa0: 0x2000001,\n\t            0xb0: 0x2100400,\n\t            0xc0: 0x2100000,\n\t            0xd0: 0x401,\n\t            0xe0: 0x100400,\n\t            0xf0: 0x2000000,\n\t            0x8: 0x2100001,\n\t            0x18: 0x0,\n\t            0x28: 0x2000401,\n\t            0x38: 0x2100400,\n\t            0x48: 0x100000,\n\t            0x58: 0x2000001,\n\t            0x68: 0x2000000,\n\t            0x78: 0x401,\n\t            0x88: 0x100401,\n\t            0x98: 0x2000400,\n\t            0xa8: 0x2100000,\n\t            0xb8: 0x100001,\n\t            0xc8: 0x400,\n\t            0xd8: 0x2100401,\n\t            0xe8: 0x1,\n\t            0xf8: 0x100400,\n\t            0x100: 0x2000000,\n\t            0x110: 0x100000,\n\t            0x120: 0x2000401,\n\t            0x130: 0x2100001,\n\t            0x140: 0x100001,\n\t            0x150: 0x2000400,\n\t            0x160: 0x2100400,\n\t            0x170: 0x100401,\n\t            0x180: 0x401,\n\t            0x190: 0x2100401,\n\t            0x1a0: 0x100400,\n\t            0x1b0: 0x1,\n\t            0x1c0: 0x0,\n\t            0x1d0: 0x2100000,\n\t            0x1e0: 0x2000001,\n\t            0x1f0: 0x400,\n\t            0x108: 0x100400,\n\t            0x118: 0x2000401,\n\t            0x128: 0x2100001,\n\t            0x138: 0x1,\n\t            0x148: 0x2000000,\n\t            0x158: 0x100000,\n\t            0x168: 0x401,\n\t            0x178: 0x2100400,\n\t            0x188: 0x2000001,\n\t            0x198: 0x2100000,\n\t            0x1a8: 0x0,\n\t            0x1b8: 0x2100401,\n\t            0x1c8: 0x100401,\n\t            0x1d8: 0x400,\n\t            0x1e8: 0x2000400,\n\t            0x1f8: 0x100001\n\t        },\n\t        {\n\t            0x0: 0x8000820,\n\t            0x1: 0x20000,\n\t            0x2: 0x8000000,\n\t            0x3: 0x20,\n\t            0x4: 0x20020,\n\t            0x5: 0x8020820,\n\t            0x6: 0x8020800,\n\t            0x7: 0x800,\n\t            0x8: 0x8020000,\n\t            0x9: 0x8000800,\n\t            0xa: 0x20800,\n\t            0xb: 0x8020020,\n\t            0xc: 0x820,\n\t            0xd: 0x0,\n\t            0xe: 0x8000020,\n\t            0xf: 0x20820,\n\t            0x80000000: 0x800,\n\t            0x80000001: 0x8020820,\n\t            0x80000002: 0x8000820,\n\t            0x80000003: 0x8000000,\n\t            0x80000004: 0x8020000,\n\t            0x80000005: 0x20800,\n\t            0x80000006: 0x20820,\n\t            0x80000007: 0x20,\n\t            0x80000008: 0x8000020,\n\t            0x80000009: 0x820,\n\t            0x8000000a: 0x20020,\n\t            0x8000000b: 0x8020800,\n\t            0x8000000c: 0x0,\n\t            0x8000000d: 0x8020020,\n\t            0x8000000e: 0x8000800,\n\t            0x8000000f: 0x20000,\n\t            0x10: 0x20820,\n\t            0x11: 0x8020800,\n\t            0x12: 0x20,\n\t            0x13: 0x800,\n\t            0x14: 0x8000800,\n\t            0x15: 0x8000020,\n\t            0x16: 0x8020020,\n\t            0x17: 0x20000,\n\t            0x18: 0x0,\n\t            0x19: 0x20020,\n\t            0x1a: 0x8020000,\n\t            0x1b: 0x8000820,\n\t            0x1c: 0x8020820,\n\t            0x1d: 0x20800,\n\t            0x1e: 0x820,\n\t            0x1f: 0x8000000,\n\t            0x80000010: 0x20000,\n\t            0x80000011: 0x800,\n\t            0x80000012: 0x8020020,\n\t            0x80000013: 0x20820,\n\t            0x80000014: 0x20,\n\t            0x80000015: 0x8020000,\n\t            0x80000016: 0x8000000,\n\t            0x80000017: 0x8000820,\n\t            0x80000018: 0x8020820,\n\t            0x80000019: 0x8000020,\n\t            0x8000001a: 0x8000800,\n\t            0x8000001b: 0x0,\n\t            0x8000001c: 0x20800,\n\t            0x8000001d: 0x820,\n\t            0x8000001e: 0x20020,\n\t            0x8000001f: 0x8020800\n\t        }\n\t    ];\n\n\t    // Masks that select the SBOX input\n\t    var SBOX_MASK = [\n\t        0xf8000001, 0x1f800000, 0x01f80000, 0x001f8000,\n\t        0x0001f800, 0x00001f80, 0x000001f8, 0x8000001f\n\t    ];\n\n\t    /**\n\t     * DES block cipher algorithm.\n\t     */\n\t    var DES = C_algo.DES = BlockCipher.extend({\n\t        _doReset: function () {\n\t            // Shortcuts\n\t            var key = this._key;\n\t            var keyWords = key.words;\n\n\t            // Select 56 bits according to PC1\n\t            var keyBits = [];\n\t            for (var i = 0; i < 56; i++) {\n\t                var keyBitPos = PC1[i] - 1;\n\t                keyBits[i] = (keyWords[keyBitPos >>> 5] >>> (31 - keyBitPos % 32)) & 1;\n\t            }\n\n\t            // Assemble 16 subkeys\n\t            var subKeys = this._subKeys = [];\n\t            for (var nSubKey = 0; nSubKey < 16; nSubKey++) {\n\t                // Create subkey\n\t                var subKey = subKeys[nSubKey] = [];\n\n\t                // Shortcut\n\t                var bitShift = BIT_SHIFTS[nSubKey];\n\n\t                // Select 48 bits according to PC2\n\t                for (var i = 0; i < 24; i++) {\n\t                    // Select from the left 28 key bits\n\t                    subKey[(i / 6) | 0] |= keyBits[((PC2[i] - 1) + bitShift) % 28] << (31 - i % 6);\n\n\t                    // Select from the right 28 key bits\n\t                    subKey[4 + ((i / 6) | 0)] |= keyBits[28 + (((PC2[i + 24] - 1) + bitShift) % 28)] << (31 - i % 6);\n\t                }\n\n\t                // Since each subkey is applied to an expanded 32-bit input,\n\t                // the subkey can be broken into 8 values scaled to 32-bits,\n\t                // which allows the key to be used without expansion\n\t                subKey[0] = (subKey[0] << 1) | (subKey[0] >>> 31);\n\t                for (var i = 1; i < 7; i++) {\n\t                    subKey[i] = subKey[i] >>> ((i - 1) * 4 + 3);\n\t                }\n\t                subKey[7] = (subKey[7] << 5) | (subKey[7] >>> 27);\n\t            }\n\n\t            // Compute inverse subkeys\n\t            var invSubKeys = this._invSubKeys = [];\n\t            for (var i = 0; i < 16; i++) {\n\t                invSubKeys[i] = subKeys[15 - i];\n\t            }\n\t        },\n\n\t        encryptBlock: function (M, offset) {\n\t            this._doCryptBlock(M, offset, this._subKeys);\n\t        },\n\n\t        decryptBlock: function (M, offset) {\n\t            this._doCryptBlock(M, offset, this._invSubKeys);\n\t        },\n\n\t        _doCryptBlock: function (M, offset, subKeys) {\n\t            // Get input\n\t            this._lBlock = M[offset];\n\t            this._rBlock = M[offset + 1];\n\n\t            // Initial permutation\n\t            exchangeLR.call(this, 4,  0x0f0f0f0f);\n\t            exchangeLR.call(this, 16, 0x0000ffff);\n\t            exchangeRL.call(this, 2,  0x33333333);\n\t            exchangeRL.call(this, 8,  0x00ff00ff);\n\t            exchangeLR.call(this, 1,  0x55555555);\n\n\t            // Rounds\n\t            for (var round = 0; round < 16; round++) {\n\t                // Shortcuts\n\t                var subKey = subKeys[round];\n\t                var lBlock = this._lBlock;\n\t                var rBlock = this._rBlock;\n\n\t                // Feistel function\n\t                var f = 0;\n\t                for (var i = 0; i < 8; i++) {\n\t                    f |= SBOX_P[i][((rBlock ^ subKey[i]) & SBOX_MASK[i]) >>> 0];\n\t                }\n\t                this._lBlock = rBlock;\n\t                this._rBlock = lBlock ^ f;\n\t            }\n\n\t            // Undo swap from last round\n\t            var t = this._lBlock;\n\t            this._lBlock = this._rBlock;\n\t            this._rBlock = t;\n\n\t            // Final permutation\n\t            exchangeLR.call(this, 1,  0x55555555);\n\t            exchangeRL.call(this, 8,  0x00ff00ff);\n\t            exchangeRL.call(this, 2,  0x33333333);\n\t            exchangeLR.call(this, 16, 0x0000ffff);\n\t            exchangeLR.call(this, 4,  0x0f0f0f0f);\n\n\t            // Set output\n\t            M[offset] = this._lBlock;\n\t            M[offset + 1] = this._rBlock;\n\t        },\n\n\t        keySize: 64/32,\n\n\t        ivSize: 64/32,\n\n\t        blockSize: 64/32\n\t    });\n\n\t    // Swap bits across the left and right words\n\t    function exchangeLR(offset, mask) {\n\t        var t = ((this._lBlock >>> offset) ^ this._rBlock) & mask;\n\t        this._rBlock ^= t;\n\t        this._lBlock ^= t << offset;\n\t    }\n\n\t    function exchangeRL(offset, mask) {\n\t        var t = ((this._rBlock >>> offset) ^ this._lBlock) & mask;\n\t        this._lBlock ^= t;\n\t        this._rBlock ^= t << offset;\n\t    }\n\n\t    /**\n\t     * Shortcut functions to the cipher's object interface.\n\t     *\n\t     * @example\n\t     *\n\t     *     var ciphertext = CryptoJS.DES.encrypt(message, key, cfg);\n\t     *     var plaintext  = CryptoJS.DES.decrypt(ciphertext, key, cfg);\n\t     */\n\t    C.DES = BlockCipher._createHelper(DES);\n\n\t    /**\n\t     * Triple-DES block cipher algorithm.\n\t     */\n\t    var TripleDES = C_algo.TripleDES = BlockCipher.extend({\n\t        _doReset: function () {\n\t            // Shortcuts\n\t            var key = this._key;\n\t            var keyWords = key.words;\n\n\t            // Create DES instances\n\t            this._des1 = DES.createEncryptor(WordArray.create(keyWords.slice(0, 2)));\n\t            this._des2 = DES.createEncryptor(WordArray.create(keyWords.slice(2, 4)));\n\t            this._des3 = DES.createEncryptor(WordArray.create(keyWords.slice(4, 6)));\n\t        },\n\n\t        encryptBlock: function (M, offset) {\n\t            this._des1.encryptBlock(M, offset);\n\t            this._des2.decryptBlock(M, offset);\n\t            this._des3.encryptBlock(M, offset);\n\t        },\n\n\t        decryptBlock: function (M, offset) {\n\t            this._des3.decryptBlock(M, offset);\n\t            this._des2.encryptBlock(M, offset);\n\t            this._des1.decryptBlock(M, offset);\n\t        },\n\n\t        keySize: 192/32,\n\n\t        ivSize: 64/32,\n\n\t        blockSize: 64/32\n\t    });\n\n\t    /**\n\t     * Shortcut functions to the cipher's object interface.\n\t     *\n\t     * @example\n\t     *\n\t     *     var ciphertext = CryptoJS.TripleDES.encrypt(message, key, cfg);\n\t     *     var plaintext  = CryptoJS.TripleDES.decrypt(ciphertext, key, cfg);\n\t     */\n\t    C.TripleDES = BlockCipher._createHelper(TripleDES);\n\t}());\n\n\n\treturn CryptoJS.TripleDES;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function (undefined) {\n\t    // Shortcuts\n\t    var C = CryptoJS;\n\t    var C_lib = C.lib;\n\t    var Base = C_lib.Base;\n\t    var X32WordArray = C_lib.WordArray;\n\n\t    /**\n\t     * x64 namespace.\n\t     */\n\t    var C_x64 = C.x64 = {};\n\n\t    /**\n\t     * A 64-bit word.\n\t     */\n\t    var X64Word = C_x64.Word = Base.extend({\n\t        /**\n\t         * Initializes a newly created 64-bit word.\n\t         *\n\t         * @param {number} high The high 32 bits.\n\t         * @param {number} low The low 32 bits.\n\t         *\n\t         * @example\n\t         *\n\t         *     var x64Word = CryptoJS.x64.Word.create(0x00010203, 0x04050607);\n\t         */\n\t        init: function (high, low) {\n\t            this.high = high;\n\t            this.low = low;\n\t        }\n\n\t        /**\n\t         * Bitwise NOTs this word.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after negating.\n\t         *\n\t         * @example\n\t         *\n\t         *     var negated = x64Word.not();\n\t         */\n\t        // not: function () {\n\t            // var high = ~this.high;\n\t            // var low = ~this.low;\n\n\t            // return X64Word.create(high, low);\n\t        // },\n\n\t        /**\n\t         * Bitwise ANDs this word with the passed word.\n\t         *\n\t         * @param {X64Word} word The x64-Word to AND with this word.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after ANDing.\n\t         *\n\t         * @example\n\t         *\n\t         *     var anded = x64Word.and(anotherX64Word);\n\t         */\n\t        // and: function (word) {\n\t            // var high = this.high & word.high;\n\t            // var low = this.low & word.low;\n\n\t            // return X64Word.create(high, low);\n\t        // },\n\n\t        /**\n\t         * Bitwise ORs this word with the passed word.\n\t         *\n\t         * @param {X64Word} word The x64-Word to OR with this word.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after ORing.\n\t         *\n\t         * @example\n\t         *\n\t         *     var ored = x64Word.or(anotherX64Word);\n\t         */\n\t        // or: function (word) {\n\t            // var high = this.high | word.high;\n\t            // var low = this.low | word.low;\n\n\t            // return X64Word.create(high, low);\n\t        // },\n\n\t        /**\n\t         * Bitwise XORs this word with the passed word.\n\t         *\n\t         * @param {X64Word} word The x64-Word to XOR with this word.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after XORing.\n\t         *\n\t         * @example\n\t         *\n\t         *     var xored = x64Word.xor(anotherX64Word);\n\t         */\n\t        // xor: function (word) {\n\t            // var high = this.high ^ word.high;\n\t            // var low = this.low ^ word.low;\n\n\t            // return X64Word.create(high, low);\n\t        // },\n\n\t        /**\n\t         * Shifts this word n bits to the left.\n\t         *\n\t         * @param {number} n The number of bits to shift.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after shifting.\n\t         *\n\t         * @example\n\t         *\n\t         *     var shifted = x64Word.shiftL(25);\n\t         */\n\t        // shiftL: function (n) {\n\t            // if (n < 32) {\n\t                // var high = (this.high << n) | (this.low >>> (32 - n));\n\t                // var low = this.low << n;\n\t            // } else {\n\t                // var high = this.low << (n - 32);\n\t                // var low = 0;\n\t            // }\n\n\t            // return X64Word.create(high, low);\n\t        // },\n\n\t        /**\n\t         * Shifts this word n bits to the right.\n\t         *\n\t         * @param {number} n The number of bits to shift.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after shifting.\n\t         *\n\t         * @example\n\t         *\n\t         *     var shifted = x64Word.shiftR(7);\n\t         */\n\t        // shiftR: function (n) {\n\t            // if (n < 32) {\n\t                // var low = (this.low >>> n) | (this.high << (32 - n));\n\t                // var high = this.high >>> n;\n\t            // } else {\n\t                // var low = this.high >>> (n - 32);\n\t                // var high = 0;\n\t            // }\n\n\t            // return X64Word.create(high, low);\n\t        // },\n\n\t        /**\n\t         * Rotates this word n bits to the left.\n\t         *\n\t         * @param {number} n The number of bits to rotate.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after rotating.\n\t         *\n\t         * @example\n\t         *\n\t         *     var rotated = x64Word.rotL(25);\n\t         */\n\t        // rotL: function (n) {\n\t            // return this.shiftL(n).or(this.shiftR(64 - n));\n\t        // },\n\n\t        /**\n\t         * Rotates this word n bits to the right.\n\t         *\n\t         * @param {number} n The number of bits to rotate.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after rotating.\n\t         *\n\t         * @example\n\t         *\n\t         *     var rotated = x64Word.rotR(7);\n\t         */\n\t        // rotR: function (n) {\n\t            // return this.shiftR(n).or(this.shiftL(64 - n));\n\t        // },\n\n\t        /**\n\t         * Adds this word with the passed word.\n\t         *\n\t         * @param {X64Word} word The x64-Word to add with this word.\n\t         *\n\t         * @return {X64Word} A new x64-Word object after adding.\n\t         *\n\t         * @example\n\t         *\n\t         *     var added = x64Word.add(anotherX64Word);\n\t         */\n\t        // add: function (word) {\n\t            // var low = (this.low + word.low) | 0;\n\t            // var carry = (low >>> 0) < (this.low >>> 0) ? 1 : 0;\n\t            // var high = (this.high + word.high + carry) | 0;\n\n\t            // return X64Word.create(high, low);\n\t        // }\n\t    });\n\n\t    /**\n\t     * An array of 64-bit words.\n\t     *\n\t     * @property {Array} words The array of CryptoJS.x64.Word objects.\n\t     * @property {number} sigBytes The number of significant bytes in this word array.\n\t     */\n\t    var X64WordArray = C_x64.WordArray = Base.extend({\n\t        /**\n\t         * Initializes a newly created word array.\n\t         *\n\t         * @param {Array} words (Optional) An array of CryptoJS.x64.Word objects.\n\t         * @param {number} sigBytes (Optional) The number of significant bytes in the words.\n\t         *\n\t         * @example\n\t         *\n\t         *     var wordArray = CryptoJS.x64.WordArray.create();\n\t         *\n\t         *     var wordArray = CryptoJS.x64.WordArray.create([\n\t         *         CryptoJS.x64.Word.create(0x00010203, 0x04050607),\n\t         *         CryptoJS.x64.Word.create(0x18191a1b, 0x1c1d1e1f)\n\t         *     ]);\n\t         *\n\t         *     var wordArray = CryptoJS.x64.WordArray.create([\n\t         *         CryptoJS.x64.Word.create(0x00010203, 0x04050607),\n\t         *         CryptoJS.x64.Word.create(0x18191a1b, 0x1c1d1e1f)\n\t         *     ], 10);\n\t         */\n\t        init: function (words, sigBytes) {\n\t            words = this.words = words || [];\n\n\t            if (sigBytes != undefined) {\n\t                this.sigBytes = sigBytes;\n\t            } else {\n\t                this.sigBytes = words.length * 8;\n\t            }\n\t        },\n\n\t        /**\n\t         * Converts this 64-bit word array to a 32-bit word array.\n\t         *\n\t         * @return {CryptoJS.lib.WordArray} This word array's data as a 32-bit word array.\n\t         *\n\t         * @example\n\t         *\n\t         *     var x32WordArray = x64WordArray.toX32();\n\t         */\n\t        toX32: function () {\n\t            // Shortcuts\n\t            var x64Words = this.words;\n\t            var x64WordsLength = x64Words.length;\n\n\t            // Convert\n\t            var x32Words = [];\n\t            for (var i = 0; i < x64WordsLength; i++) {\n\t                var x64Word = x64Words[i];\n\t                x32Words.push(x64Word.high);\n\t                x32Words.push(x64Word.low);\n\t            }\n\n\t            return X32WordArray.create(x32Words, this.sigBytes);\n\t        },\n\n\t        /**\n\t         * Creates a copy of this word array.\n\t         *\n\t         * @return {X64WordArray} The clone.\n\t         *\n\t         * @example\n\t         *\n\t         *     var clone = x64WordArray.clone();\n\t         */\n\t        clone: function () {\n\t            var clone = Base.clone.call(this);\n\n\t            // Clone \"words\" array\n\t            var words = clone.words = this.words.slice(0);\n\n\t            // Clone each X64Word object\n\t            var wordsLength = words.length;\n\t            for (var i = 0; i < wordsLength; i++) {\n\t                words[i] = words[i].clone();\n\t            }\n\n\t            return clone;\n\t        }\n\t    });\n\t}());\n\n\n\treturn CryptoJS;\n\n}));","(function (self) {\n  'use strict';\n\n  function fetchPonyfill(options) {\n    var Promise = options && options.Promise || self.Promise;\n    var XMLHttpRequest = options && options.XMLHttpRequest || self.XMLHttpRequest;\n    var global = self;\n\n    return (function () {\n      var self = Object.create(global, {\n        fetch: {\n          value: undefined,\n          writable: true\n        }\n      });\n\n      (function(self) {\n        'use strict';\n\n        if (self.fetch) {\n          return\n        }\n\n        var support = {\n          searchParams: 'URLSearchParams' in self,\n          iterable: 'Symbol' in self && 'iterator' in Symbol,\n          blob: 'FileReader' in self && 'Blob' in self && (function() {\n            try {\n              new Blob()\n              return true\n            } catch(e) {\n              return false\n            }\n          })(),\n          formData: 'FormData' in self,\n          arrayBuffer: 'ArrayBuffer' in self\n        }\n\n        if (support.arrayBuffer) {\n          var viewClasses = [\n            '[object Int8Array]',\n            '[object Uint8Array]',\n            '[object Uint8ClampedArray]',\n            '[object Int16Array]',\n            '[object Uint16Array]',\n            '[object Int32Array]',\n            '[object Uint32Array]',\n            '[object Float32Array]',\n            '[object Float64Array]'\n          ]\n\n          var isDataView = function(obj) {\n            return obj && DataView.prototype.isPrototypeOf(obj)\n          }\n\n          var isArrayBufferView = ArrayBuffer.isView || function(obj) {\n            return obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1\n          }\n        }\n\n        function normalizeName(name) {\n          if (typeof name !== 'string') {\n            name = String(name)\n          }\n          if (/[^a-z0-9\\-#$%&'*+.\\^_`|~]/i.test(name)) {\n            throw new TypeError('Invalid character in header field name')\n          }\n          return name.toLowerCase()\n        }\n\n        function normalizeValue(value) {\n          if (typeof value !== 'string') {\n            value = String(value)\n          }\n          return value\n        }\n\n        // Build a destructive iterator for the value list\n        function iteratorFor(items) {\n          var iterator = {\n            next: function() {\n              var value = items.shift()\n              return {done: value === undefined, value: value}\n            }\n          }\n\n          if (support.iterable) {\n            iterator[Symbol.iterator] = function() {\n              return iterator\n            }\n          }\n\n          return iterator\n        }\n\n        function Headers(headers) {\n          this.map = {}\n\n          if (headers instanceof Headers) {\n            headers.forEach(function(value, name) {\n              this.append(name, value)\n            }, this)\n          } else if (Array.isArray(headers)) {\n            headers.forEach(function(header) {\n              this.append(header[0], header[1])\n            }, this)\n          } else if (headers) {\n            Object.getOwnPropertyNames(headers).forEach(function(name) {\n              this.append(name, headers[name])\n            }, this)\n          }\n        }\n\n        Headers.prototype.append = function(name, value) {\n          name = normalizeName(name)\n          value = normalizeValue(value)\n          var oldValue = this.map[name]\n          this.map[name] = oldValue ? oldValue+','+value : value\n        }\n\n        Headers.prototype['delete'] = function(name) {\n          delete this.map[normalizeName(name)]\n        }\n\n        Headers.prototype.get = function(name) {\n          name = normalizeName(name)\n          return this.has(name) ? this.map[name] : null\n        }\n\n        Headers.prototype.has = function(name) {\n          return this.map.hasOwnProperty(normalizeName(name))\n        }\n\n        Headers.prototype.set = function(name, value) {\n          this.map[normalizeName(name)] = normalizeValue(value)\n        }\n\n        Headers.prototype.forEach = function(callback, thisArg) {\n          for (var name in this.map) {\n            if (this.map.hasOwnProperty(name)) {\n              callback.call(thisArg, this.map[name], name, this)\n            }\n          }\n        }\n\n        Headers.prototype.keys = function() {\n          var items = []\n          this.forEach(function(value, name) { items.push(name) })\n          return iteratorFor(items)\n        }\n\n        Headers.prototype.values = function() {\n          var items = []\n          this.forEach(function(value) { items.push(value) })\n          return iteratorFor(items)\n        }\n\n        Headers.prototype.entries = function() {\n          var items = []\n          this.forEach(function(value, name) { items.push([name, value]) })\n          return iteratorFor(items)\n        }\n\n        if (support.iterable) {\n          Headers.prototype[Symbol.iterator] = Headers.prototype.entries\n        }\n\n        function consumed(body) {\n          if (body.bodyUsed) {\n            return Promise.reject(new TypeError('Already read'))\n          }\n          body.bodyUsed = true\n        }\n\n        function fileReaderReady(reader) {\n          return new Promise(function(resolve, reject) {\n            reader.onload = function() {\n              resolve(reader.result)\n            }\n            reader.onerror = function() {\n              reject(reader.error)\n            }\n          })\n        }\n\n        function readBlobAsArrayBuffer(blob) {\n          var reader = new FileReader()\n          var promise = fileReaderReady(reader)\n          reader.readAsArrayBuffer(blob)\n          return promise\n        }\n\n        function readBlobAsText(blob) {\n          var reader = new FileReader()\n          var promise = fileReaderReady(reader)\n          reader.readAsText(blob)\n          return promise\n        }\n\n        function readArrayBufferAsText(buf) {\n          var view = new Uint8Array(buf)\n          var chars = new Array(view.length)\n\n          for (var i = 0; i < view.length; i++) {\n            chars[i] = String.fromCharCode(view[i])\n          }\n          return chars.join('')\n        }\n\n        function bufferClone(buf) {\n          if (buf.slice) {\n            return buf.slice(0)\n          } else {\n            var view = new Uint8Array(buf.byteLength)\n            view.set(new Uint8Array(buf))\n            return view.buffer\n          }\n        }\n\n        function Body() {\n          this.bodyUsed = false\n\n          this._initBody = function(body) {\n            this._bodyInit = body\n            if (!body) {\n              this._bodyText = ''\n            } else if (typeof body === 'string') {\n              this._bodyText = body\n            } else if (support.blob && Blob.prototype.isPrototypeOf(body)) {\n              this._bodyBlob = body\n            } else if (support.formData && FormData.prototype.isPrototypeOf(body)) {\n              this._bodyFormData = body\n            } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n              this._bodyText = body.toString()\n            } else if (support.arrayBuffer && support.blob && isDataView(body)) {\n              this._bodyArrayBuffer = bufferClone(body.buffer)\n              // IE 10-11 can't handle a DataView body.\n              this._bodyInit = new Blob([this._bodyArrayBuffer])\n            } else if (support.arrayBuffer && (ArrayBuffer.prototype.isPrototypeOf(body) || isArrayBufferView(body))) {\n              this._bodyArrayBuffer = bufferClone(body)\n            } else {\n              throw new Error('unsupported BodyInit type')\n            }\n\n            if (!this.headers.get('content-type')) {\n              if (typeof body === 'string') {\n                this.headers.set('content-type', 'text/plain;charset=UTF-8')\n              } else if (this._bodyBlob && this._bodyBlob.type) {\n                this.headers.set('content-type', this._bodyBlob.type)\n              } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n                this.headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8')\n              }\n            }\n          }\n\n          if (support.blob) {\n            this.blob = function() {\n              var rejected = consumed(this)\n              if (rejected) {\n                return rejected\n              }\n\n              if (this._bodyBlob) {\n                return Promise.resolve(this._bodyBlob)\n              } else if (this._bodyArrayBuffer) {\n                return Promise.resolve(new Blob([this._bodyArrayBuffer]))\n              } else if (this._bodyFormData) {\n                throw new Error('could not read FormData body as blob')\n              } else {\n                return Promise.resolve(new Blob([this._bodyText]))\n              }\n            }\n\n            this.arrayBuffer = function() {\n              if (this._bodyArrayBuffer) {\n                return consumed(this) || Promise.resolve(this._bodyArrayBuffer)\n              } else {\n                return this.blob().then(readBlobAsArrayBuffer)\n              }\n            }\n          }\n\n          this.text = function() {\n            var rejected = consumed(this)\n            if (rejected) {\n              return rejected\n            }\n\n            if (this._bodyBlob) {\n              return readBlobAsText(this._bodyBlob)\n            } else if (this._bodyArrayBuffer) {\n              return Promise.resolve(readArrayBufferAsText(this._bodyArrayBuffer))\n            } else if (this._bodyFormData) {\n              throw new Error('could not read FormData body as text')\n            } else {\n              return Promise.resolve(this._bodyText)\n            }\n          }\n\n          if (support.formData) {\n            this.formData = function() {\n              return this.text().then(decode)\n            }\n          }\n\n          this.json = function() {\n            return this.text().then(JSON.parse)\n          }\n\n          return this\n        }\n\n        // HTTP methods whose capitalization should be normalized\n        var methods = ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'POST', 'PUT']\n\n        function normalizeMethod(method) {\n          var upcased = method.toUpperCase()\n          return (methods.indexOf(upcased) > -1) ? upcased : method\n        }\n\n        function Request(input, options) {\n          options = options || {}\n          var body = options.body\n\n          if (input instanceof Request) {\n            if (input.bodyUsed) {\n              throw new TypeError('Already read')\n            }\n            this.url = input.url\n            this.credentials = input.credentials\n            if (!options.headers) {\n              this.headers = new Headers(input.headers)\n            }\n            this.method = input.method\n            this.mode = input.mode\n            if (!body && input._bodyInit != null) {\n              body = input._bodyInit\n              input.bodyUsed = true\n            }\n          } else {\n            this.url = String(input)\n          }\n\n          this.credentials = options.credentials || this.credentials || 'omit'\n          if (options.headers || !this.headers) {\n            this.headers = new Headers(options.headers)\n          }\n          this.method = normalizeMethod(options.method || this.method || 'GET')\n          this.mode = options.mode || this.mode || null\n          this.referrer = null\n\n          if ((this.method === 'GET' || this.method === 'HEAD') && body) {\n            throw new TypeError('Body not allowed for GET or HEAD requests')\n          }\n          this._initBody(body)\n        }\n\n        Request.prototype.clone = function() {\n          return new Request(this, { body: this._bodyInit })\n        }\n\n        function decode(body) {\n          var form = new FormData()\n          body.trim().split('&').forEach(function(bytes) {\n            if (bytes) {\n              var split = bytes.split('=')\n              var name = split.shift().replace(/\\+/g, ' ')\n              var value = split.join('=').replace(/\\+/g, ' ')\n              form.append(decodeURIComponent(name), decodeURIComponent(value))\n            }\n          })\n          return form\n        }\n\n        function parseHeaders(rawHeaders) {\n          var headers = new Headers()\n          rawHeaders.split(/\\r?\\n/).forEach(function(line) {\n            var parts = line.split(':')\n            var key = parts.shift().trim()\n            if (key) {\n              var value = parts.join(':').trim()\n              headers.append(key, value)\n            }\n          })\n          return headers\n        }\n\n        Body.call(Request.prototype)\n\n        function Response(bodyInit, options) {\n          if (!options) {\n            options = {}\n          }\n\n          this.type = 'default'\n          this.status = 'status' in options ? options.status : 200\n          this.ok = this.status >= 200 && this.status < 300\n          this.statusText = 'statusText' in options ? options.statusText : 'OK'\n          this.headers = new Headers(options.headers)\n          this.url = options.url || ''\n          this._initBody(bodyInit)\n        }\n\n        Body.call(Response.prototype)\n\n        Response.prototype.clone = function() {\n          return new Response(this._bodyInit, {\n            status: this.status,\n            statusText: this.statusText,\n            headers: new Headers(this.headers),\n            url: this.url\n          })\n        }\n\n        Response.error = function() {\n          var response = new Response(null, {status: 0, statusText: ''})\n          response.type = 'error'\n          return response\n        }\n\n        var redirectStatuses = [301, 302, 303, 307, 308]\n\n        Response.redirect = function(url, status) {\n          if (redirectStatuses.indexOf(status) === -1) {\n            throw new RangeError('Invalid status code')\n          }\n\n          return new Response(null, {status: status, headers: {location: url}})\n        }\n\n        self.Headers = Headers\n        self.Request = Request\n        self.Response = Response\n\n        self.fetch = function(input, init) {\n          return new Promise(function(resolve, reject) {\n            var request = new Request(input, init)\n            var xhr = new XMLHttpRequest()\n\n            xhr.onload = function() {\n              var options = {\n                status: xhr.status,\n                statusText: xhr.statusText,\n                headers: parseHeaders(xhr.getAllResponseHeaders() || '')\n              }\n              options.url = 'responseURL' in xhr ? xhr.responseURL : options.headers.get('X-Request-URL')\n              var body = 'response' in xhr ? xhr.response : xhr.responseText\n              resolve(new Response(body, options))\n            }\n\n            xhr.onerror = function() {\n              reject(new TypeError('Network request failed'))\n            }\n\n            xhr.ontimeout = function() {\n              reject(new TypeError('Network request failed'))\n            }\n\n            xhr.open(request.method, request.url, true)\n\n            if (request.credentials === 'include') {\n              xhr.withCredentials = true\n            }\n\n            if ('responseType' in xhr && support.blob) {\n              xhr.responseType = 'blob'\n            }\n\n            request.headers.forEach(function(value, name) {\n              xhr.setRequestHeader(name, value)\n            })\n\n            xhr.send(typeof request._bodyInit === 'undefined' ? null : request._bodyInit)\n          })\n        }\n        self.fetch.polyfill = true\n      })(typeof self !== 'undefined' ? self : this);\n\n\n      return {\n        fetch: self.fetch,\n        Headers: self.Headers,\n        Request: self.Request,\n        Response: self.Response\n      };\n    }());\n  }\n\n  if (typeof define === 'function' && define.amd) {\n    define(function () {\n      return fetchPonyfill;\n    });\n  } else if (typeof exports === 'object') {\n    module.exports = fetchPonyfill;\n  } else {\n    self.fetchPonyfill = fetchPonyfill;\n  }\n}(typeof self === 'undefined' ? this : self));\n\n","// shim for using process in browser\nvar process = module.exports = {};\n\n// cached from whatever global is present so that test runners that stub it\n// don't break things.  But we need to wrap it in a try catch in case it is\n// wrapped in strict mode code which doesn't define any globals.  It's inside a\n// function because try/catches deoptimize in certain engines.\n\nvar cachedSetTimeout;\nvar cachedClearTimeout;\n\nfunction defaultSetTimout() {\n    throw new Error('setTimeout has not been defined');\n}\nfunction defaultClearTimeout () {\n    throw new Error('clearTimeout has not been defined');\n}\n(function () {\n    try {\n        if (typeof setTimeout === 'function') {\n            cachedSetTimeout = setTimeout;\n        } else {\n            cachedSetTimeout = defaultSetTimout;\n        }\n    } catch (e) {\n        cachedSetTimeout = defaultSetTimout;\n    }\n    try {\n        if (typeof clearTimeout === 'function') {\n            cachedClearTimeout = clearTimeout;\n        } else {\n            cachedClearTimeout = defaultClearTimeout;\n        }\n    } catch (e) {\n        cachedClearTimeout = defaultClearTimeout;\n    }\n} ())\nfunction runTimeout(fun) {\n    if (cachedSetTimeout === setTimeout) {\n        //normal enviroments in sane situations\n        return setTimeout(fun, 0);\n    }\n    // if setTimeout wasn't available but was latter defined\n    if ((cachedSetTimeout === defaultSetTimout || !cachedSetTimeout) && setTimeout) {\n        cachedSetTimeout = setTimeout;\n        return setTimeout(fun, 0);\n    }\n    try {\n        // when when somebody has screwed with setTimeout but no I.E. maddness\n        return cachedSetTimeout(fun, 0);\n    } catch(e){\n        try {\n            // When we are in I.E. but the script has been evaled so I.E. doesn't trust the global object when called normally\n            return cachedSetTimeout.call(null, fun, 0);\n        } catch(e){\n            // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error\n            return cachedSetTimeout.call(this, fun, 0);\n        }\n    }\n\n\n}\nfunction runClearTimeout(marker) {\n    if (cachedClearTimeout === clearTimeout) {\n        //normal enviroments in sane situations\n        return clearTimeout(marker);\n    }\n    // if clearTimeout wasn't available but was latter defined\n    if ((cachedClearTimeout === defaultClearTimeout || !cachedClearTimeout) && clearTimeout) {\n        cachedClearTimeout = clearTimeout;\n        return clearTimeout(marker);\n    }\n    try {\n        // when when somebody has screwed with setTimeout but no I.E. maddness\n        return cachedClearTimeout(marker);\n    } catch (e){\n        try {\n            // When we are in I.E. but the script has been evaled so I.E. doesn't  trust the global object when called normally\n            return cachedClearTimeout.call(null, marker);\n        } catch (e){\n            // same as above but when it's a version of I.E. that must have the global object for 'this', hopfully our context correct otherwise it will throw a global error.\n            // Some versions of I.E. have different rules for clearTimeout vs setTimeout\n            return cachedClearTimeout.call(this, marker);\n        }\n    }\n\n\n\n}\nvar queue = [];\nvar draining = false;\nvar currentQueue;\nvar queueIndex = -1;\n\nfunction cleanUpNextTick() {\n    if (!draining || !currentQueue) {\n        return;\n    }\n    draining = false;\n    if (currentQueue.length) {\n        queue = currentQueue.concat(queue);\n    } else {\n        queueIndex = -1;\n    }\n    if (queue.length) {\n        drainQueue();\n    }\n}\n\nfunction drainQueue() {\n    if (draining) {\n        return;\n    }\n    var timeout = runTimeout(cleanUpNextTick);\n    draining = true;\n\n    var len = queue.length;\n    while(len) {\n        currentQueue = queue;\n        queue = [];\n        while (++queueIndex < len) {\n            if (currentQueue) {\n                currentQueue[queueIndex].run();\n            }\n        }\n        queueIndex = -1;\n        len = queue.length;\n    }\n    currentQueue = null;\n    draining = false;\n    runClearTimeout(timeout);\n}\n\nprocess.nextTick = function (fun) {\n    var args = new Array(arguments.length - 1);\n    if (arguments.length > 1) {\n        for (var i = 1; i < arguments.length; i++) {\n            args[i - 1] = arguments[i];\n        }\n    }\n    queue.push(new Item(fun, args));\n    if (queue.length === 1 && !draining) {\n        runTimeout(drainQueue);\n    }\n};\n\n// v8 likes predictible objects\nfunction Item(fun, array) {\n    this.fun = fun;\n    this.array = array;\n}\nItem.prototype.run = function () {\n    this.fun.apply(null, this.array);\n};\nprocess.title = 'browser';\nprocess.browser = true;\nprocess.env = {};\nprocess.argv = [];\nprocess.version = ''; // empty string to avoid regexp issues\nprocess.versions = {};\n\nfunction noop() {}\n\nprocess.on = noop;\nprocess.addListener = noop;\nprocess.once = noop;\nprocess.off = noop;\nprocess.removeListener = noop;\nprocess.removeAllListeners = noop;\nprocess.emit = noop;\nprocess.prependListener = noop;\nprocess.prependOnceListener = noop;\n\nprocess.listeners = function (name) { return [] }\n\nprocess.binding = function (name) {\n    throw new Error('process.binding is not supported');\n};\n\nprocess.cwd = function () { return '/' };\nprocess.chdir = function (dir) {\n    throw new Error('process.chdir is not supported');\n};\nprocess.umask = function() { return 0; };\n","'use strict';\n\nvar replace = String.prototype.replace;\nvar percentTwenties = /%20/g;\n\nmodule.exports = {\n    'default': 'RFC3986',\n    formatters: {\n        RFC1738: function (value) {\n            return replace.call(value, percentTwenties, '+');\n        },\n        RFC3986: function (value) {\n            return value;\n        }\n    },\n    RFC1738: 'RFC1738',\n    RFC3986: 'RFC3986'\n};\n","'use strict';\n\nvar stringify = require('./stringify');\nvar parse = require('./parse');\nvar formats = require('./formats');\n\nmodule.exports = {\n    formats: formats,\n    parse: parse,\n    stringify: stringify\n};\n","'use strict';\n\nvar utils = require('./utils');\n\nvar has = Object.prototype.hasOwnProperty;\n\nvar defaults = {\n    allowDots: false,\n    allowPrototypes: false,\n    arrayLimit: 20,\n    decoder: utils.decode,\n    delimiter: '&',\n    depth: 5,\n    parameterLimit: 1000,\n    plainObjects: false,\n    strictNullHandling: false\n};\n\nvar parseValues = function parseQueryStringValues(str, options) {\n    var obj = {};\n    var cleanStr = options.ignoreQueryPrefix ? str.replace(/^\\?/, '') : str;\n    var limit = options.parameterLimit === Infinity ? undefined : options.parameterLimit;\n    var parts = cleanStr.split(options.delimiter, limit);\n\n    for (var i = 0; i < parts.length; ++i) {\n        var part = parts[i];\n\n        var bracketEqualsPos = part.indexOf(']=');\n        var pos = bracketEqualsPos === -1 ? part.indexOf('=') : bracketEqualsPos + 1;\n\n        var key, val;\n        if (pos === -1) {\n            key = options.decoder(part, defaults.decoder);\n            val = options.strictNullHandling ? null : '';\n        } else {\n            key = options.decoder(part.slice(0, pos), defaults.decoder);\n            val = options.decoder(part.slice(pos + 1), defaults.decoder);\n        }\n        if (has.call(obj, key)) {\n            obj[key] = [].concat(obj[key]).concat(val);\n        } else {\n            obj[key] = val;\n        }\n    }\n\n    return obj;\n};\n\nvar parseObject = function (chain, val, options) {\n    var leaf = val;\n\n    for (var i = chain.length - 1; i >= 0; --i) {\n        var obj;\n        var root = chain[i];\n\n        if (root === '[]') {\n            obj = [];\n            obj = obj.concat(leaf);\n        } else {\n            obj = options.plainObjects ? Object.create(null) : {};\n            var cleanRoot = root.charAt(0) === '[' && root.charAt(root.length - 1) === ']' ? root.slice(1, -1) : root;\n            var index = parseInt(cleanRoot, 10);\n            if (\n                !isNaN(index)\n                && root !== cleanRoot\n                && String(index) === cleanRoot\n                && index >= 0\n                && (options.parseArrays && index <= options.arrayLimit)\n            ) {\n                obj = [];\n                obj[index] = leaf;\n            } else {\n                obj[cleanRoot] = leaf;\n            }\n        }\n\n        leaf = obj;\n    }\n\n    return leaf;\n};\n\nvar parseKeys = function parseQueryStringKeys(givenKey, val, options) {\n    if (!givenKey) {\n        return;\n    }\n\n    // Transform dot notation to bracket notation\n    var key = options.allowDots ? givenKey.replace(/\\.([^.[]+)/g, '[$1]') : givenKey;\n\n    // The regex chunks\n\n    var brackets = /(\\[[^[\\]]*])/;\n    var child = /(\\[[^[\\]]*])/g;\n\n    // Get the parent\n\n    var segment = brackets.exec(key);\n    var parent = segment ? key.slice(0, segment.index) : key;\n\n    // Stash the parent if it exists\n\n    var keys = [];\n    if (parent) {\n        // If we aren't using plain objects, optionally prefix keys\n        // that would overwrite object prototype properties\n        if (!options.plainObjects && has.call(Object.prototype, parent)) {\n            if (!options.allowPrototypes) {\n                return;\n            }\n        }\n\n        keys.push(parent);\n    }\n\n    // Loop through children appending to the array until we hit depth\n\n    var i = 0;\n    while ((segment = child.exec(key)) !== null && i < options.depth) {\n        i += 1;\n        if (!options.plainObjects && has.call(Object.prototype, segment[1].slice(1, -1))) {\n            if (!options.allowPrototypes) {\n                return;\n            }\n        }\n        keys.push(segment[1]);\n    }\n\n    // If there's a remainder, just add whatever is left\n\n    if (segment) {\n        keys.push('[' + key.slice(segment.index) + ']');\n    }\n\n    return parseObject(keys, val, options);\n};\n\nmodule.exports = function (str, opts) {\n    var options = opts ? utils.assign({}, opts) : {};\n\n    if (options.decoder !== null && options.decoder !== undefined && typeof options.decoder !== 'function') {\n        throw new TypeError('Decoder has to be a function.');\n    }\n\n    options.ignoreQueryPrefix = options.ignoreQueryPrefix === true;\n    options.delimiter = typeof options.delimiter === 'string' || utils.isRegExp(options.delimiter) ? options.delimiter : defaults.delimiter;\n    options.depth = typeof options.depth === 'number' ? options.depth : defaults.depth;\n    options.arrayLimit = typeof options.arrayLimit === 'number' ? options.arrayLimit : defaults.arrayLimit;\n    options.parseArrays = options.parseArrays !== false;\n    options.decoder = typeof options.decoder === 'function' ? options.decoder : defaults.decoder;\n    options.allowDots = typeof options.allowDots === 'boolean' ? options.allowDots : defaults.allowDots;\n    options.plainObjects = typeof options.plainObjects === 'boolean' ? options.plainObjects : defaults.plainObjects;\n    options.allowPrototypes = typeof options.allowPrototypes === 'boolean' ? options.allowPrototypes : defaults.allowPrototypes;\n    options.parameterLimit = typeof options.parameterLimit === 'number' ? options.parameterLimit : defaults.parameterLimit;\n    options.strictNullHandling = typeof options.strictNullHandling === 'boolean' ? options.strictNullHandling : defaults.strictNullHandling;\n\n    if (str === '' || str === null || typeof str === 'undefined') {\n        return options.plainObjects ? Object.create(null) : {};\n    }\n\n    var tempObj = typeof str === 'string' ? parseValues(str, options) : str;\n    var obj = options.plainObjects ? Object.create(null) : {};\n\n    // Iterate over the keys and setup the new object\n\n    var keys = Object.keys(tempObj);\n    for (var i = 0; i < keys.length; ++i) {\n        var key = keys[i];\n        var newObj = parseKeys(key, tempObj[key], options);\n        obj = utils.merge(obj, newObj, options);\n    }\n\n    return utils.compact(obj);\n};\n","'use strict';\n\nvar utils = require('./utils');\nvar formats = require('./formats');\n\nvar arrayPrefixGenerators = {\n    brackets: function brackets(prefix) { // eslint-disable-line func-name-matching\n        return prefix + '[]';\n    },\n    indices: function indices(prefix, key) { // eslint-disable-line func-name-matching\n        return prefix + '[' + key + ']';\n    },\n    repeat: function repeat(prefix) { // eslint-disable-line func-name-matching\n        return prefix;\n    }\n};\n\nvar toISO = Date.prototype.toISOString;\n\nvar defaults = {\n    delimiter: '&',\n    encode: true,\n    encoder: utils.encode,\n    encodeValuesOnly: false,\n    serializeDate: function serializeDate(date) { // eslint-disable-line func-name-matching\n        return toISO.call(date);\n    },\n    skipNulls: false,\n    strictNullHandling: false\n};\n\nvar stringify = function stringify( // eslint-disable-line func-name-matching\n    object,\n    prefix,\n    generateArrayPrefix,\n    strictNullHandling,\n    skipNulls,\n    encoder,\n    filter,\n    sort,\n    allowDots,\n    serializeDate,\n    formatter,\n    encodeValuesOnly\n) {\n    var obj = object;\n    if (typeof filter === 'function') {\n        obj = filter(prefix, obj);\n    } else if (obj instanceof Date) {\n        obj = serializeDate(obj);\n    } else if (obj === null) {\n        if (strictNullHandling) {\n            return encoder && !encodeValuesOnly ? encoder(prefix, defaults.encoder) : prefix;\n        }\n\n        obj = '';\n    }\n\n    if (typeof obj === 'string' || typeof obj === 'number' || typeof obj === 'boolean' || utils.isBuffer(obj)) {\n        if (encoder) {\n            var keyValue = encodeValuesOnly ? prefix : encoder(prefix, defaults.encoder);\n            return [formatter(keyValue) + '=' + formatter(encoder(obj, defaults.encoder))];\n        }\n        return [formatter(prefix) + '=' + formatter(String(obj))];\n    }\n\n    var values = [];\n\n    if (typeof obj === 'undefined') {\n        return values;\n    }\n\n    var objKeys;\n    if (Array.isArray(filter)) {\n        objKeys = filter;\n    } else {\n        var keys = Object.keys(obj);\n        objKeys = sort ? keys.sort(sort) : keys;\n    }\n\n    for (var i = 0; i < objKeys.length; ++i) {\n        var key = objKeys[i];\n\n        if (skipNulls && obj[key] === null) {\n            continue;\n        }\n\n        if (Array.isArray(obj)) {\n            values = values.concat(stringify(\n                obj[key],\n                generateArrayPrefix(prefix, key),\n                generateArrayPrefix,\n                strictNullHandling,\n                skipNulls,\n                encoder,\n                filter,\n                sort,\n                allowDots,\n                serializeDate,\n                formatter,\n                encodeValuesOnly\n            ));\n        } else {\n            values = values.concat(stringify(\n                obj[key],\n                prefix + (allowDots ? '.' + key : '[' + key + ']'),\n                generateArrayPrefix,\n                strictNullHandling,\n                skipNulls,\n                encoder,\n                filter,\n                sort,\n                allowDots,\n                serializeDate,\n                formatter,\n                encodeValuesOnly\n            ));\n        }\n    }\n\n    return values;\n};\n\nmodule.exports = function (object, opts) {\n    var obj = object;\n    var options = opts ? utils.assign({}, opts) : {};\n\n    if (options.encoder !== null && options.encoder !== undefined && typeof options.encoder !== 'function') {\n        throw new TypeError('Encoder has to be a function.');\n    }\n\n    var delimiter = typeof options.delimiter === 'undefined' ? defaults.delimiter : options.delimiter;\n    var strictNullHandling = typeof options.strictNullHandling === 'boolean' ? options.strictNullHandling : defaults.strictNullHandling;\n    var skipNulls = typeof options.skipNulls === 'boolean' ? options.skipNulls : defaults.skipNulls;\n    var encode = typeof options.encode === 'boolean' ? options.encode : defaults.encode;\n    var encoder = typeof options.encoder === 'function' ? options.encoder : defaults.encoder;\n    var sort = typeof options.sort === 'function' ? options.sort : null;\n    var allowDots = typeof options.allowDots === 'undefined' ? false : options.allowDots;\n    var serializeDate = typeof options.serializeDate === 'function' ? options.serializeDate : defaults.serializeDate;\n    var encodeValuesOnly = typeof options.encodeValuesOnly === 'boolean' ? options.encodeValuesOnly : defaults.encodeValuesOnly;\n    if (typeof options.format === 'undefined') {\n        options.format = formats['default'];\n    } else if (!Object.prototype.hasOwnProperty.call(formats.formatters, options.format)) {\n        throw new TypeError('Unknown format option provided.');\n    }\n    var formatter = formats.formatters[options.format];\n    var objKeys;\n    var filter;\n\n    if (typeof options.filter === 'function') {\n        filter = options.filter;\n        obj = filter('', obj);\n    } else if (Array.isArray(options.filter)) {\n        filter = options.filter;\n        objKeys = filter;\n    }\n\n    var keys = [];\n\n    if (typeof obj !== 'object' || obj === null) {\n        return '';\n    }\n\n    var arrayFormat;\n    if (options.arrayFormat in arrayPrefixGenerators) {\n        arrayFormat = options.arrayFormat;\n    } else if ('indices' in options) {\n        arrayFormat = options.indices ? 'indices' : 'repeat';\n    } else {\n        arrayFormat = 'indices';\n    }\n\n    var generateArrayPrefix = arrayPrefixGenerators[arrayFormat];\n\n    if (!objKeys) {\n        objKeys = Object.keys(obj);\n    }\n\n    if (sort) {\n        objKeys.sort(sort);\n    }\n\n    for (var i = 0; i < objKeys.length; ++i) {\n        var key = objKeys[i];\n\n        if (skipNulls && obj[key] === null) {\n            continue;\n        }\n\n        keys = keys.concat(stringify(\n            obj[key],\n            key,\n            generateArrayPrefix,\n            strictNullHandling,\n            skipNulls,\n            encode ? encoder : null,\n            filter,\n            sort,\n            allowDots,\n            serializeDate,\n            formatter,\n            encodeValuesOnly\n        ));\n    }\n\n    var joined = keys.join(delimiter);\n    var prefix = options.addQueryPrefix === true ? '?' : '';\n\n    return joined.length > 0 ? prefix + joined : '';\n};\n","'use strict';\n\nvar has = Object.prototype.hasOwnProperty;\n\nvar hexTable = (function () {\n    var array = [];\n    for (var i = 0; i < 256; ++i) {\n        array.push('%' + ((i < 16 ? '0' : '') + i.toString(16)).toUpperCase());\n    }\n\n    return array;\n}());\n\nvar compactQueue = function compactQueue(queue) {\n    var obj;\n\n    while (queue.length) {\n        var item = queue.pop();\n        obj = item.obj[item.prop];\n\n        if (Array.isArray(obj)) {\n            var compacted = [];\n\n            for (var j = 0; j < obj.length; ++j) {\n                if (typeof obj[j] !== 'undefined') {\n                    compacted.push(obj[j]);\n                }\n            }\n\n            item.obj[item.prop] = compacted;\n        }\n    }\n\n    return obj;\n};\n\nexports.arrayToObject = function arrayToObject(source, options) {\n    var obj = options && options.plainObjects ? Object.create(null) : {};\n    for (var i = 0; i < source.length; ++i) {\n        if (typeof source[i] !== 'undefined') {\n            obj[i] = source[i];\n        }\n    }\n\n    return obj;\n};\n\nexports.merge = function merge(target, source, options) {\n    if (!source) {\n        return target;\n    }\n\n    if (typeof source !== 'object') {\n        if (Array.isArray(target)) {\n            target.push(source);\n        } else if (typeof target === 'object') {\n            if (options.plainObjects || options.allowPrototypes || !has.call(Object.prototype, source)) {\n                target[source] = true;\n            }\n        } else {\n            return [target, source];\n        }\n\n        return target;\n    }\n\n    if (typeof target !== 'object') {\n        return [target].concat(source);\n    }\n\n    var mergeTarget = target;\n    if (Array.isArray(target) && !Array.isArray(source)) {\n        mergeTarget = exports.arrayToObject(target, options);\n    }\n\n    if (Array.isArray(target) && Array.isArray(source)) {\n        source.forEach(function (item, i) {\n            if (has.call(target, i)) {\n                if (target[i] && typeof target[i] === 'object') {\n                    target[i] = exports.merge(target[i], item, options);\n                } else {\n                    target.push(item);\n                }\n            } else {\n                target[i] = item;\n            }\n        });\n        return target;\n    }\n\n    return Object.keys(source).reduce(function (acc, key) {\n        var value = source[key];\n\n        if (has.call(acc, key)) {\n            acc[key] = exports.merge(acc[key], value, options);\n        } else {\n            acc[key] = value;\n        }\n        return acc;\n    }, mergeTarget);\n};\n\nexports.assign = function assignSingleSource(target, source) {\n    return Object.keys(source).reduce(function (acc, key) {\n        acc[key] = source[key];\n        return acc;\n    }, target);\n};\n\nexports.decode = function (str) {\n    try {\n        return decodeURIComponent(str.replace(/\\+/g, ' '));\n    } catch (e) {\n        return str;\n    }\n};\n\nexports.encode = function encode(str) {\n    // This code was originally written by Brian White (mscdex) for the io.js core querystring library.\n    // It has been adapted here for stricter adherence to RFC 3986\n    if (str.length === 0) {\n        return str;\n    }\n\n    var string = typeof str === 'string' ? str : String(str);\n\n    var out = '';\n    for (var i = 0; i < string.length; ++i) {\n        var c = string.charCodeAt(i);\n\n        if (\n            c === 0x2D // -\n            || c === 0x2E // .\n            || c === 0x5F // _\n            || c === 0x7E // ~\n            || (c >= 0x30 && c <= 0x39) // 0-9\n            || (c >= 0x41 && c <= 0x5A) // a-z\n            || (c >= 0x61 && c <= 0x7A) // A-Z\n        ) {\n            out += string.charAt(i);\n            continue;\n        }\n\n        if (c < 0x80) {\n            out = out + hexTable[c];\n            continue;\n        }\n\n        if (c < 0x800) {\n            out = out + (hexTable[0xC0 | (c >> 6)] + hexTable[0x80 | (c & 0x3F)]);\n            continue;\n        }\n\n        if (c < 0xD800 || c >= 0xE000) {\n            out = out + (hexTable[0xE0 | (c >> 12)] + hexTable[0x80 | ((c >> 6) & 0x3F)] + hexTable[0x80 | (c & 0x3F)]);\n            continue;\n        }\n\n        i += 1;\n        c = 0x10000 + (((c & 0x3FF) << 10) | (string.charCodeAt(i) & 0x3FF));\n        out += hexTable[0xF0 | (c >> 18)]\n            + hexTable[0x80 | ((c >> 12) & 0x3F)]\n            + hexTable[0x80 | ((c >> 6) & 0x3F)]\n            + hexTable[0x80 | (c & 0x3F)];\n    }\n\n    return out;\n};\n\nexports.compact = function compact(value) {\n    var queue = [{ obj: { o: value }, prop: 'o' }];\n    var refs = [];\n\n    for (var i = 0; i < queue.length; ++i) {\n        var item = queue[i];\n        var obj = item.obj[item.prop];\n\n        var keys = Object.keys(obj);\n        for (var j = 0; j < keys.length; ++j) {\n            var key = keys[j];\n            var val = obj[key];\n            if (typeof val === 'object' && val !== null && refs.indexOf(val) === -1) {\n                queue.push({ obj: obj, prop: key });\n                refs.push(val);\n            }\n        }\n    }\n\n    return compactQueue(queue);\n};\n\nexports.isRegExp = function isRegExp(obj) {\n    return Object.prototype.toString.call(obj) === '[object RegExp]';\n};\n\nexports.isBuffer = function isBuffer(obj) {\n    if (obj === null || typeof obj === 'undefined') {\n        return false;\n    }\n\n    return !!(obj.constructor && obj.constructor.isBuffer && obj.constructor.isBuffer(obj));\n};\n"]} diff --git a/doc/manual.rst b/doc/manual.rst index fd8418fe4df76..680f73930438d 100644 --- a/doc/manual.rst +++ b/doc/manual.rst @@ -340,9 +340,11 @@ Here's an overview of base exchange properties with values added for example: }, 'version': 'v1', // string ending with digits 'api': { ... }, // dictionary of api endpoints - 'hasFetchTickers': true, // true if the exchange implements fetchTickers () - 'hasFetchOHLCV': false, // true if the exchange implements fetchOHLCV () - 'timeframes': { // empty if the exchange !hasFetchOHLCV + 'has': { + 'fetchTickers': true, // true if the exchange implements fetchTickers () + 'fetchOHLCV': false, // true if the exchange implements fetchOHLCV () + }, + 'timeframes': { // empty if the exchange !has.fetchOHLCV '1m': '1minute', '1h': '1hour', '1d': '1day', @@ -1157,7 +1159,7 @@ You can call the unified ``fetchOHLCV`` / ``fetch_ohlcv`` method to get the list // JavaScript let sleep = (ms) => new Promise (resolve => setTimeout (resolve, ms)); - if (exchange.hasFetchOHLCV) { + if (exchange.has.fetchOHLCV) { (async () => { for (symbol in exchange.markets) { await sleep (exchange.rateLimit) // milliseconds diff --git a/examples/js/fetch-ticker-where-available.js b/examples/js/fetch-ticker-where-available.js index a31484fe7bdc8..5bb157f6219ec 100644 --- a/examples/js/fetch-ticker-where-available.js +++ b/examples/js/fetch-ticker-where-available.js @@ -21,7 +21,7 @@ let printUsage = function () { let id = ccxt.exchanges[i] const exchange = new ccxt[id] () - if (exchange.hasPublicAPI) { + if (exchange.has.publicAPI) { try { diff --git a/examples/js/search-all-exchanges.js b/examples/js/search-all-exchanges.js index 008297f9f88e4..051d4b8f37f4c 100644 --- a/examples/js/search-all-exchanges.js +++ b/examples/js/search-all-exchanges.js @@ -51,7 +51,7 @@ const checkAgainst = strict ? // instantiate the exchange let exchange = new ccxt[id] () - if (exchange.hasPublicAPI) { + if (exchange.has.publicAPI) { try { diff --git a/js/base/Exchange.js b/js/base/Exchange.js index 4a03f1d6cdd1c..62e99cf5b5c09 100644 --- a/js/base/Exchange.js +++ b/js/base/Exchange.js @@ -2,14 +2,7 @@ /* ------------------------------------------------------------------------ */ -<<<<<<< HEAD const functions = require ('./functions') -======= -const isNode = (typeof window === 'undefined') && !(typeof WorkerGlobalScope !== 'undefined' && self instanceof WorkerGlobalScope) - , functions = require ('./functions') - , throttle = require ('./throttle') - , defaultFetch = isNode ? require ('fetch-ponyfill')().fetch : fetch ->>>>>>> master , Market = require ('./Market') const { isNode @@ -28,9 +21,9 @@ const { isNode , throttle , capitalize } = functions -const { sleep +const { now + , sleep , timeout - , time , TimedOut } = require ('./functions/time') const { ExchangeError @@ -40,7 +33,7 @@ const { ExchangeError , RequestTimeout , ExchangeNotAvailable } = require ('./errors') -const fetchImplementation = isNode ? require ('fetch-ponyfill')().fetch : fetch +const defaultFetch = isNode ? require ('fetch-ponyfill')().fetch : fetch const journal = undefined // isNode && require ('./journal') // stub until we get a better solution for Webpack and React @@ -95,9 +88,9 @@ module.exports = class Exchange { this.iso8601 = timestamp => new Date (timestamp).toISOString () this.parse8601 = x => Date.parse (((x.indexOf ('+') >= 0) || (x.slice (-1) == 'Z')) ? x : (x + 'Z')) - this.milliseconds = () => Math.floor (time.now ()) - this.microseconds = () => Math.floor (time.now () * 1000) - this.seconds = () => Math.floor (time.now () / 1000) + this.milliseconds = now + this.microseconds = () => now () * 1000 // TODO: utilize performance.now for that purpose + this.seconds = () => Math.floor (now () / 1000) this.id = undefined // rate limiter settings diff --git a/js/base/functions.js b/js/base/functions.js index 8a2fb9ff79670..f9d0cc9eb11f7 100644 --- a/js/base/functions.js +++ b/js/base/functions.js @@ -11,38 +11,18 @@ const unCamelCasePropertyNames = x => { /* ------------------------------------------------------------------------ */ -module.exports = unCamelCasePropertyNames ({ - - ...require ('./functions/platform'), - ...require ('./functions/generic'), - ...require ('./functions/string'), - ...require ('./functions/type'), - ...require ('./functions/number'), - ...require ('./functions/encode'), - ...require ('./functions/crypto'), - ...require ('./functions/time'), - ...require ('./functions/throttle'), - -/* ............................................. */ - - json: JSON.stringify, - unjson: JSON.parse, - -/* ............................................. */ - - aggregate (bidasks) { - - let result = {} - - for (const [price, volume] of bidasks) { - if (volume > 0) - result[price] = (result[price] || 0) + volume - } - - return Object.keys (result) - .map (price => [parseFloat (price), - parseFloat (result[price])]) - } -}) +module.exports = unCamelCasePropertyNames (Object.assign ({} + + , require ('./functions/platform') + , require ('./functions/generic') + , require ('./functions/string') + , require ('./functions/type') + , require ('./functions/number') + , require ('./functions/encode') + , require ('./functions/crypto') + , require ('./functions/time') + , require ('./functions/throttle') + , require ('./functions/misc') +)) /* ------------------------------------------------------------------------ */ diff --git a/js/base/functions/encode.js b/js/base/functions/encode.js index 621d791d80df4..7ad3d30fe11a5 100644 --- a/js/base/functions/encode.js +++ b/js/base/functions/encode.js @@ -2,15 +2,19 @@ /* ------------------------------------------------------------------------ */ -const qs = require ('qs') // querystring (TODO: get rid of that dependency) +const CryptoJS = require ('crypto-js') +const qs = require ('qs') // querystring (TODO: get rid of that dependency) /* ------------------------------------------------------------------------ */ -module.exports = { - - stringToBinary (str) { +module.exports = + + { json: JSON.stringify + , unjson: JSON.parse + + , stringToBinary (str) { const arr = new Uint8Array (str.length) - for (let i = 0; i < str.length; i++) { arr[i] = str.charCodeAt(i); } + for (let i = 0; i < str.length; i++) { arr[i] = str.charCodeAt (i); } return CryptoJS.lib.WordArray.create (arr) } diff --git a/js/base/functions/misc.js b/js/base/functions/misc.js new file mode 100644 index 0000000000000..d858a8eeadc15 --- /dev/null +++ b/js/base/functions/misc.js @@ -0,0 +1,22 @@ +"use strict"; + +/* ------------------------------------------------------------------------ */ + +module.exports = { + + aggregate (bidasks) { + + let result = {} + + for (const [price, volume] of bidasks) { + if (volume > 0) + result[price] = (result[price] || 0) + volume + } + + return Object.keys (result) + .map (price => [parseFloat (price), + parseFloat (result[price])]) + } +} + +/* ------------------------------------------------------------------------ */ diff --git a/js/base/functions/number.js b/js/base/functions/number.js index df3d67870db99..9a5d55629b6ac 100644 --- a/js/base/functions/number.js +++ b/js/base/functions/number.js @@ -1,18 +1,9 @@ -"use strict"; - -const { isString, isNumber } = require ('./type') - -/* ------------------------------------------------------------------------ */ - -const decimal = float => parseFloat (float).toString () - -/* ------------------------------------------------------------------------ */ +/* NB: A LEGACY CODE, WILL BE RE-WRITTEN VERY SOON + ------------------------------------------------------------------------ */ // See https://stackoverflow.com/questions/1685680/how-to-avoid-scientific-notation-for-large-numbers-in-javascript for discussion -function numberToString (x) { // avoid scientific notation for too large and too small numbers - - if (isString (x)) return x +function toFixed (x) { // avoid scientific notation for too large and too small numbers if (Math.abs (x) < 1.0) { const e = parseInt (x.toString ().split ('e-')[1]) @@ -28,110 +19,173 @@ function numberToString (x) { // avoid scientific notation for too large and too x += (new Array (e+1)).join ('0') } } - return x.toString () + return x } -/* ------------------------------------------------------------------------ */ +// See https://stackoverflow.com/questions/4912788/truncate-not-round-off-decimal-numbers-in-javascript for discussion -const padWithZeroes = (x, digits = 0) => { +// > So, after all it turned out, rounding bugs will always haunt you, no matter how hard you try to compensate them. +// > Hence the problem should be attacked by representing numbers exactly in decimal notation. + +const truncate_regExpCache = [] + , truncate_to_string = (num, precision = 0) => { + num = toFixed (num) + if (precision > 0) { + const re = truncate_regExpCache[precision] || (truncate_regExpCache[precision] = new RegExp("([-]*\\d+\\.\\d{" + precision + "})(\\d)")) + const [,result] = num.toString ().match (re) || [null, num] + return result.toString () + } + return parseInt (num).toString () + } + , truncate = (num, precision = 0) => parseFloat (truncate_to_string (num, precision)) + +const precisionFromString = (string) => { + const split = string.replace (/0+$/g, '').split ('.') + return (split.length > 1) ? (split[1].length) : 0 +} - const [int, frac = ''] = x.split ('.') +module.exports = { - return int + ((frac || (digits > 0)) - ? ('.' + frac.padEnd (digits, '0')) - : '') + toFixed, + truncate_to_string, + truncate, + precisionFromString } -/* ------------------------------------------------------------------------ */ +// "use strict"; -const roundDecimalString = (s, to, afterDot = false) => { +// const { isString, isNumber } = require ('./type') - const digits = Array.from (s) - const result = [] - const dot = s.indexOf ('.') +// /* ------------------------------------------------------------------------ */ - let memo = 0 +// const decimal = float => parseFloat (float).toString () - if (afterDot) to = ((dot >= 0) ? dot : digits.length) + to +// /* ------------------------------------------------------------------------ */ - for (let i = digits.length - 1; i >= 0; i--) { - const d = digits[i] - if (d !== '.') { - let n = (d.charCodeAt (0) - 48) + memo - let numDigitsAhead = i - let dotAhead = (dot >= 0) && (i >= dot) - if ((numDigitsAhead + (dotAhead ? -1 : 0)) >= to) { // ignore dot when counting digits ahead - n = (n > 5) ? 10 : 0 // rounding on per-digit basis - } - if (n > 9) { n = 0; memo = 1; } - else memo = 0 - digits[i] = n - } - } - return (memo || '') + digits.join ('') -} +// // See https://stackoverflow.com/questions/1685680/how-to-avoid-scientific-notation-for-large-numbers-in-javascript for discussion + +// function numberToString (x) { // avoid scientific notation for too large and too small numbers + +// if (isString (x)) return x + +// if (Math.abs (x) < 1.0) { +// const e = parseInt (x.toString ().split ('e-')[1]) +// if (e) { +// x *= Math.pow (10, e-1) +// x = '0.' + (new Array (e)).join ('0') + x.toString ().substring (2) +// } +// } else { +// let e = parseInt (x.toString ().split ('+')[1]) +// if (e > 20) { +// e -= 20 +// x /= Math.pow (10, e) +// x += (new Array (e+1)).join ('0') +// } +// } +// return x.toString () +// } + +// /* ------------------------------------------------------------------------ */ + +// const padWithZeroes = (x, digits = 0) => { -/* ------------------------------------------------------------------------ */ +// const [int, frac = ''] = x.split ('.') -const roundNumber = (x, { digits = 8, fixed = true }) => { // accepts either strings or Numbers +// return int + ((frac || (digits > 0)) +// ? ('.' + frac.padEnd (digits, '0')) +// : '') +// } + +// /* ------------------------------------------------------------------------ */ + +// const roundDecimalString = (s, to, afterDot = false) => { + +// const digits = Array.from (s) +// const result = [] +// const dot = s.indexOf ('.') + +// let memo = 0 + +// if (afterDot) to = ((dot >= 0) ? dot : digits.length) + to + +// for (let i = digits.length - 1; i >= 0; i--) { +// const d = digits[i] +// if (d !== '.') { +// let n = (d.charCodeAt (0) - 48) + memo +// let numDigitsAhead = i +// let dotAhead = (dot >= 0) && (i >= dot) +// if ((numDigitsAhead + (dotAhead ? -1 : 0)) >= to) { // ignore dot when counting digits ahead +// n = (n > 5) ? 10 : 0 // rounding on per-digit basis +// } +// if (n > 9) { n = 0; memo = 1; } +// else memo = 0 +// digits[i] = n +// } +// } +// return (memo || '') + digits.join ('') +// } + +// /* ------------------------------------------------------------------------ */ + +// const roundNumber = (x, { digits = 8, fixed = true }) => { // accepts either strings or Numbers - const s = numberToString (x) +// const s = numberToString (x) - if (fixed) { - return roundDecimalString (s, digits, true) +// if (fixed) { +// return roundDecimalString (s, digits, true) - } else { - const [,zeros,significantPart] = s.match (/^([^1-9]*)(.+)$/) - return zeros + roundDecimalString (significantPart, digits) - } -} +// } else { +// const [,zeros,significantPart] = s.match (/^([^1-9]*)(.+)$/) +// return zeros + roundDecimalString (significantPart, digits) +// } +// } -/* ------------------------------------------------------------------------ */ +// /* ------------------------------------------------------------------------ */ -// See https://stackoverflow.com/questions/4912788/truncate-not-round-off-decimal-numbers-in-javascript for discussion +// // See https://stackoverflow.com/questions/4912788/truncate-not-round-off-decimal-numbers-in-javascript for discussion -// > So, after all it turned out, rounding bugs will always haunt you, no matter how hard you try to compensate them. -// > Hence the problem should be attacked by representing numbers exactly in decimal notation. +// // > So, after all it turned out, rounding bugs will always haunt you, no matter how hard you try to compensate them. +// // > Hence the problem should be attacked by representing numbers exactly in decimal notation. -const regexCache = [] -const truncNumber = (x, { digits = 0, fixed = true }) => { // accepts either strings or Numbers +// const regexCache = [] +// const truncNumber = (x, { digits = 0, fixed = true }) => { // accepts either strings or Numbers - const s = numberToString (x) +// const s = numberToString (x) - if (digits > 0) { - const re = regexCache[digits] || (regexCache[digits] = new RegExp("([-]*\\d+\\.\\d{" + digits + "})(\\d)")) - const [,result] = s.match (re) || [null, s] - return fixed - ? padWithZeroes (result, digits) - : result +// if (digits > 0) { +// const re = regexCache[digits] || (regexCache[digits] = new RegExp("([-]*\\d+\\.\\d{" + digits + "})(\\d)")) +// const [,result] = s.match (re) || [null, s] +// return fixed +// ? padWithZeroes (result, digits) +// : result - } else { - throw new Error ('not implemented yet') - } -} +// } else { +// throw new Error ('not implemented yet') +// } +// } -/* ------------------------------------------------------------------------ */ +// /* ------------------------------------------------------------------------ */ -const precisionFromString = (string) => { - const split = string.replace (/0+$/g, '').split ('.') - return (split.length > 1) ? (split[1].length) : 0 -} +// const precisionFromString = (string) => { +// const split = string.replace (/0+$/g, '').split ('.') +// return (split.length > 1) ? (split[1].length) : 0 +// } -/* ------------------------------------------------------------------------ */ +// /* ------------------------------------------------------------------------ */ -const toPrecision = (x, { round = true, digits = 8, fixed = true }) => round ? roundNumber (x, { digits, fixed }) - : truncNumber (x, { digits, fixed }) +// const toPrecision = (x, { round = true, digits = 8, fixed = true }) => round ? roundNumber (x, { digits, fixed }) +// : truncNumber (x, { digits, fixed }) -/* ------------------------------------------------------------------------ */ +// /* ------------------------------------------------------------------------ */ -module.exports = { +// module.exports = { - decimal, - numberToString, - toPrecision, - padWithZeroes, - roundDecimalString, - precisionFromString -} - -/* ------------------------------------------------------------------------ */ +// decimal, +// numberToString, +// toPrecision, +// padWithZeroes, +// roundDecimalString, +// precisionFromString +// } + +// /* ------------------------------------------------------------------------ */ diff --git a/js/base/functions/throttle.js b/js/base/functions/throttle.js index 417e66e40b749..848c376b54562 100644 --- a/js/base/functions/throttle.js +++ b/js/base/functions/throttle.js @@ -3,7 +3,7 @@ /* ------------------------------------------------------------------------ */ const { sleep - , time } = require ('./time') + , now } = require ('./time') /* ------------------------------------------------------------------------ */ @@ -11,7 +11,7 @@ module.exports = { throttle: function throttle (cfg) { - let lastTimestamp = time.now () + let lastTimestamp = now () , numTokens = (typeof cfg.numTokens != 'undefined') ? cfg.numTokens : cfg.capacity , running = false , counter = 0 @@ -43,9 +43,9 @@ module.exports = { } } } - const now = time.now () - , elapsed = now - lastTimestamp - lastTimestamp = now + const t = now () + , elapsed = t - lastTimestamp + lastTimestamp = t numTokens = Math.min (cfg.capacity, numTokens + elapsed * cfg.refillRate) await sleep (cfg.delay) } diff --git a/js/base/functions/time.js b/js/base/functions/time.js index 333a8c67d06fa..e7d7e5d3e8464 100644 --- a/js/base/functions/time.js +++ b/js/base/functions/time.js @@ -6,14 +6,12 @@ const { isNode } = require ('./platform') /* ------------------------------------------------------------------------ */ -const time = isNode ? (require ('perf_hooks').performance) : // a built-in high-resolution timer available in Node - (typeof performance !== 'undefined') ? performance // ...it is also should be available in modern browsers - : Date // (fall back to the default standard resolution timer) +const now = Date.now // TODO: figure out how to utilize performance.now () properly – it's not as easy as it does not return a unix timestamp... /* ------------------------------------------------------------------------ */ const setTimeout_original = setTimeout -const setTimeout_safe = (done, ms, setTimeout = setTimeout_original /* overrideable for mocking purposes */, targetTime = time.now () + ms) => { +const setTimeout_safe = (done, ms, setTimeout = setTimeout_original /* overrideable for mocking purposes */, targetTime = now () + ms) => { /* The built-in setTimeout function can fire its callback earlier than specified, so we need to ensure that it does not happen: sleep recursively until `targetTime` is reached... */ @@ -23,7 +21,7 @@ const setTimeout_safe = (done, ms, setTimeout = setTimeout_original /* overridea let id = setTimeout (() => { active = true - const rest = targetTime - time.now () + const rest = targetTime - now () if (rest > 0) { clearInnerTimeout = setTimeout_safe (done, rest, setTimeout, targetTime) // try sleep more } else { @@ -47,8 +45,8 @@ class TimedOut extends Error { constructor () { const message = 'timed out' super (message) - this.constructor = Error - this.__proto__ = Error.prototype + this.constructor = TimedOut + this.__proto__ = TimedOut.prototype this.message = message } } @@ -57,7 +55,7 @@ class TimedOut extends Error { module.exports = - { time + { now , setTimeout_safe , sleep: ms => new Promise (resolve => setTimeout_safe (resolve, ms)) , TimedOut diff --git a/js/okcoinusd.js b/js/okcoinusd.js index 2910ce9890c53..8ffcc68bb4aa4 100644 --- a/js/okcoinusd.js +++ b/js/okcoinusd.js @@ -24,9 +24,9 @@ module.exports = class okcoinusd extends Exchange { 'fetchOpenOrders': true, 'fetchClosedOrders': true, 'withdraw': true, + 'futureMarkets': false, }, 'extension': '.do', // appended to endpoint URL - 'hasFutureMarkets': false, 'timeframes': { '1m': '1min', '3m': '3min', @@ -184,7 +184,7 @@ module.exports = class okcoinusd extends Exchange { }, }); result.push (market); - if ((this.hasFutureMarkets) && (market['quote'] === 'USDT')) { + if ((this.has['futureMarkets']) && (market['quote'] === 'USDT')) { result.push (this.extend (market, { 'quote': 'USD', 'symbol': market['base'] + '/USD', diff --git a/js/test/test.js b/js/test/test.js index 7346c85e0acb6..f90fcf4ce22c6 100644 --- a/js/test/test.js +++ b/js/test/test.js @@ -348,7 +348,7 @@ let testFetchCurrencies = async (exchange, symbol) => { let testInvalidOrder = async (exchange, symbol) => { - if (!exchange.hasCreateOrder) { + if (!exchange.has.createOrder) { log ('order creation not supported'); return; } @@ -371,7 +371,7 @@ let testInvalidOrder = async (exchange, symbol) => { let testInsufficientFunds = async (exchange, symbol) => { - if (!exchange.hasCreateOrder) { + if (!exchange.has.createOrder) { log ('order creation not supported'); return; } diff --git a/js/test/test_base.js b/js/test/test_base.js index c70795400b6f2..cd3cd115139d5 100644 --- a/js/test/test_base.js +++ b/js/test/test_base.js @@ -1,3 +1,5 @@ +'use strict' + /* ------------------------------------------------------------------------ */ global.log = require ('ololog') // for easier debugging @@ -6,11 +8,11 @@ global.log = require ('ololog') // for easier debugging const ccxt = require ('../../ccxt.js') , assert = require ('assert') - , ansi = require ('ansicolor').nice; + , ansi = require ('ansicolor').nice /* ------------------------------------------------------------------------ */ -const { keys, values, unique, index } = ccxt +const { now, keys, values, unique, index } = ccxt /* ------------------------------------------------------------------------ */ @@ -55,14 +57,14 @@ describe ('ccxt base code', () => { assert.strictEqual (ccxt.safeInteger ({'x': 1.59999999}, 'x'), 1) }) - it ('setTimeout_safe is working', (done) => { + // TODO: make a more robust test that is not failing on certain machines under certain conditions + it.skip ('setTimeout_safe is working', (done) => { - const time = ccxt.time - const start = time.now () + const start = now () const calls = [] const brokenSetTimeout = (done, ms) => { - calls.push ({ when: time.now () - start, ms_asked: ms }) + calls.push ({ when: now () - start, ms_asked: ms }) return setTimeout (done, 100) // simulates a defect setTimeout implementation that sleeps wrong time (100ms always in this test) } @@ -156,6 +158,7 @@ describe ('ccxt base code', () => { assert.equal (exchange.amountToLots ('ETH/USD', 1.123), '1.1') }) + // TODO: make a more robust test that is not failing on certain machines under certain conditions it.skip ('rate limiting works', async () => { const calls = [] @@ -172,7 +175,7 @@ describe ('ccxt base code', () => { tokenBucket: { capacity, numTokens, defaultCost, delay }, async ping (...args) { return this.throttle ().then (() => exchange.pong (...args)) }, - async pong (...args) { calls.push ({ when: ccxt.time.now (), path: args[0], args }) } + async pong (...args) { calls.push ({ when: now (), path: args[0], args }) } }) await exchange.ping ('foo') @@ -489,7 +492,8 @@ describe ('ccxt base code', () => { assert (exchange.hasFetchDepositAddress === true) }) - it.only ('padWithZeros (from number.js) works', () => { + // TODO + it.skip ('padWithZeros (from number.js) works', () => { const { padWithZeroes } = require ('../base/functions/number') @@ -501,7 +505,8 @@ describe ('ccxt base code', () => { assert.strictEqual (padWithZeroes ('123.4567', 10), '123.4567000000') }) - it.only ('number.js works', () => { + // TODO + it.skip ('number.js works', () => { const { toPrecision, truncate, roundDecimalString } = require ('../base/functions/number') diff --git a/package.json b/package.json index 6f4c56c72b2f4..3b6926ae8f83d 100644 --- a/package.json +++ b/package.json @@ -42,7 +42,7 @@ "devDependencies": { "ansicolor": "^1.1.71", "as-table": "^1.0.31", - "asciichart": "^1.5.1", + "asciichart": "^1.5.7", "browserify": "^14.5.0", "chai": "^4.1.2", "chai-as-promised": "^7.1.1", @@ -50,7 +50,7 @@ "eslint": "^4.15.0", "istanbul": "^0.4.5", "mocha": "^3.5.3", - "nyc": "^11.0.3", + "nyc": "^11.4.1", "ololog": "^1.1.84" }, "author": { diff --git a/php/Exchange.php b/php/Exchange.php index a9cca71336bc3..e1af650ac54ff 100644 --- a/php/Exchange.php +++ b/php/Exchange.php @@ -542,26 +542,6 @@ public function __construct ($options = array ()) { $this->timeframes = null; $this->parseJsonResponse = true; - $this->hasPublicAPI = true; - $this->hasPrivateAPI = true; - $this->hasCORS = false; - $this->hasDeposit = false; - $this->hasFetchBalance = true; - $this->hasFetchClosedOrders = false; - $this->hasFetchCurrencies = false; - $this->hasFetchMyTrades = false; - $this->hasFetchOHLCV = false; - $this->hasFetchOpenOrders = false; - $this->hasFetchOrder = false; - $this->hasFetchOrderBook = true; - $this->hasFetchOrders = false; - $this->hasFetchTicker = true; - $this->hasFetchTickers = false; - $this->hasFetchTrades = true; - $this->hasWithdraw = false; - $this->hasCreateOrder = $this->hasPrivateAPI; - $this->hasCancelOrder = $this->hasPrivateAPI; - $this->requiredCredentials = array ( 'apiKey' => true, 'secret' => true, @@ -572,9 +552,12 @@ public function __construct ($options = array ()) { // API methods metainfo $this->has = array ( - 'cancelOrder' => $this->hasPrivateAPI, + 'CORS' => true, + 'privateAPI' => true, + 'publicAPI' => true, + 'cancelOrder' => true, 'createDepositAddress' => false, - 'createOrder' => $this->hasPrivateAPI, + 'createOrder' => true, 'deposit' => false, 'fetchBalance' => true, 'fetchClosedOrders' => false, diff --git a/php/_1broker.php b/php/_1broker.php index 000af833a1776..bb79197610c72 100644 --- a/php/_1broker.php +++ b/php/_1broker.php @@ -11,10 +11,12 @@ public function describe () { 'countries' => 'US', 'rateLimit' => 1500, 'version' => 'v2', - 'hasPublicAPI' => false, - 'hasCORS' => true, - 'hasFetchTrades' => false, - 'hasFetchOHLCV' => true, + 'has' => array ( + 'publicAPI' => false, + 'CORS' => true, + 'fetchTrades' => false, + 'fetchOHLCV' => true, + ), 'timeframes' => array ( '1m' => '60', '15m' => '900', diff --git a/php/_1btcxe.php b/php/_1btcxe.php index 274c8318fdcd0..5e613b4541aa8 100644 --- a/php/_1btcxe.php +++ b/php/_1btcxe.php @@ -10,9 +10,11 @@ public function describe () { 'name' => '1BTCXE', 'countries' => 'PA', // Panama 'comment' => 'Crypto Capital API', - 'hasCORS' => true, - 'hasFetchOHLCV' => true, - 'hasWithdraw' => true, + 'has' => array ( + 'CORS' => true, + 'fetchTickers' => true, + 'withdraw' => true, + ), 'timeframes' => array ( '1d' => '1year', ), diff --git a/php/acx.php b/php/acx.php index 3a4f38c94b640..6de604ae089e2 100644 --- a/php/acx.php +++ b/php/acx.php @@ -11,10 +11,12 @@ public function describe () { 'countries' => 'AU', 'rateLimit' => 1000, 'version' => 'v2', - 'hasCORS' => true, - 'hasFetchTickers' => true, - 'hasFetchOHLCV' => true, - 'hasWithdraw' => true, + 'has' => array ( + 'CORS' => true, + 'fetchTickers' => true, + 'fetchOHLCV' => true, + 'withdraw' => true, + ), 'timeframes' => array ( '1m' => '1', '5m' => '5', diff --git a/php/allcoin.php b/php/allcoin.php index 801beb485de91..5eebea918dae8 100644 --- a/php/allcoin.php +++ b/php/allcoin.php @@ -9,7 +9,9 @@ public function describe () { 'id' => 'allcoin', 'name' => 'Allcoin', 'countries' => 'CA', - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'extension' => '', 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/31561809-c316b37c-b061-11e7-8d5a-b547b4d730eb.jpg', diff --git a/php/anxpro.php b/php/anxpro.php index f9dd67a4c628d..a6e3f91aee651 100644 --- a/php/anxpro.php +++ b/php/anxpro.php @@ -11,9 +11,11 @@ public function describe () { 'countries' => array ( 'JP', 'SG', 'HK', 'NZ' ), 'version' => '2', 'rateLimit' => 1500, - 'hasCORS' => false, - 'hasFetchTrades' => false, - 'hasWithdraw' => true, + 'has' => array ( + 'CORS' => false, + 'fetchTrades' => false, + 'withdraw' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27765983-fd8595da-5ec9-11e7-82e3-adb3ab8c2612.jpg', 'api' => 'https://anxpro.com/api', diff --git a/php/bibox.php b/php/bibox.php index 47524763ed89b..48b2b6088b846 100644 --- a/php/bibox.php +++ b/php/bibox.php @@ -10,16 +10,9 @@ public function describe () { 'name' => 'Bibox', 'countries' => array ( 'CN', 'US', 'KR' ), 'version' => 'v1', - 'hasCORS' => false, - 'hasPublicAPI' => false, - 'hasFetchBalance' => true, - 'hasFetchCurrencies' => true, - 'hasFetchTickers' => true, - 'hasFetchOrders' => true, - 'hasFetchMyTrades' => true, - 'hasFetchOHLCV' => true, - 'hasWithdraw' => true, 'has' => array ( + 'CORS' => false, + 'publicAPI' => false, 'fetchBalance' => true, 'fetchCurrencies' => true, 'fetchTickers' => true, diff --git a/php/binance.php b/php/binance.php index 279f932a8680b..77db276890a35 100644 --- a/php/binance.php +++ b/php/binance.php @@ -10,18 +10,9 @@ public function describe () { 'name' => 'Binance', 'countries' => 'JP', // Japan 'rateLimit' => 500, - 'hasCORS' => false, - // obsolete metainfo interface - 'hasFetchBidsAsks' => true, - 'hasFetchTickers' => true, - 'hasFetchOHLCV' => true, - 'hasFetchMyTrades' => true, - 'hasFetchOrder' => true, - 'hasFetchOrders' => true, - 'hasFetchOpenOrders' => true, - 'hasWithdraw' => true, // new metainfo interface 'has' => array ( + 'CORS' => false, 'fetchBidsAsks' => true, 'fetchTickers' => true, 'fetchOHLCV' => true, diff --git a/php/bit2c.php b/php/bit2c.php index 017bd29a85f9b..73bfbceb5f51f 100644 --- a/php/bit2c.php +++ b/php/bit2c.php @@ -10,7 +10,9 @@ public function describe () { 'name' => 'Bit2C', 'countries' => 'IL', // Israel 'rateLimit' => 3000, - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766119-3593220e-5ece-11e7-8b3a-5a041f6bcc3f.jpg', 'api' => 'https://www.bit2c.co.il', diff --git a/php/bitbay.php b/php/bitbay.php index 8be78b6e7d428..f93fa8e9c55ec 100644 --- a/php/bitbay.php +++ b/php/bitbay.php @@ -10,8 +10,10 @@ public function describe () { 'name' => 'BitBay', 'countries' => array ( 'PL', 'EU' ), // Poland 'rateLimit' => 1000, - 'hasCORS' => true, - 'hasWithdraw' => true, + 'has' => array ( + 'CORS' => true, + 'withdraw' => true + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766132-978a7bd8-5ece-11e7-9540-bc96d1e9bbb8.jpg', 'www' => 'https://bitbay.net', diff --git a/php/bitcoincoid.php b/php/bitcoincoid.php index 4d27634eeb504..f465494ca2c96 100644 --- a/php/bitcoincoid.php +++ b/php/bitcoincoid.php @@ -9,19 +9,8 @@ public function describe () { 'id' => 'bitcoincoid', 'name' => 'Bitcoin.co.id', 'countries' => 'ID', // Indonesia - 'hasCORS' => false, - // obsolete metainfo interface - 'hasFetchTickers' => false, - 'hasFetchOHLCV' => false, - 'hasFetchOrder' => true, - 'hasFetchOrders' => false, - 'hasFetchClosedOrders' => true, - 'hasFetchOpenOrders' => true, - 'hasFetchMyTrades' => false, - 'hasFetchCurrencies' => false, - 'hasWithdraw' => false, - // new metainfo interface 'has' => array ( + 'CORS' => false, 'fetchTickers' => false, 'fetchOHLCV' => false, 'fetchOrder' => true, diff --git a/php/bitfinex.php b/php/bitfinex.php index 9f5132928df4a..c7ac2ed19004b 100644 --- a/php/bitfinex.php +++ b/php/bitfinex.php @@ -11,17 +11,9 @@ public function describe () { 'countries' => 'VG', 'version' => 'v1', 'rateLimit' => 1500, - 'hasCORS' => false, - // old metainfo interface - 'hasFetchOrder' => true, - 'hasFetchTickers' => true, - 'hasDeposit' => true, - 'hasWithdraw' => true, - 'hasFetchOHLCV' => true, - 'hasFetchOpenOrders' => true, - 'hasFetchClosedOrders' => true, // new metainfo interface 'has' => array ( + 'CORS' => false, 'fetchOHLCV' => true, 'fetchTickers' => true, 'fetchOrder' => true, diff --git a/php/bitfinex2.php b/php/bitfinex2.php index 4d39423e12cc9..9bdfe456d3016 100644 --- a/php/bitfinex2.php +++ b/php/bitfinex2.php @@ -10,18 +10,9 @@ public function describe () { 'name' => 'Bitfinex v2', 'countries' => 'VG', 'version' => 'v2', - 'hasCORS' => true, - // old metainfo interface - 'hasCreateOrder' => false, - 'hasFetchOrder' => true, - 'hasFetchTickers' => true, - 'hasFetchOHLCV' => true, - 'hasWithdraw' => true, - 'hasDeposit' => false, - 'hasFetchOpenOrders' => false, - 'hasFetchClosedOrders' => false, // new metainfo interface 'has' => array ( + 'CORS' => true, 'createOrder' => false, 'fetchOHLCV' => true, 'fetchTickers' => true, diff --git a/php/bitflyer.php b/php/bitflyer.php index b7faaf9f8673c..ef310c16c1468 100644 --- a/php/bitflyer.php +++ b/php/bitflyer.php @@ -11,8 +11,10 @@ public function describe () { 'countries' => 'JP', 'version' => 'v1', 'rateLimit' => 500, - 'hasCORS' => false, - 'hasWithdraw' => true, + 'has' => array ( + 'CORS' => false, + 'withdraw' => true + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/28051642-56154182-660e-11e7-9b0d-6042d1e6edd8.jpg', 'api' => 'https://api.bitflyer.jp', diff --git a/php/bithumb.php b/php/bithumb.php index f6cb39b71d37e..6c3bbc3f05923 100644 --- a/php/bithumb.php +++ b/php/bithumb.php @@ -10,12 +10,8 @@ public function describe () { 'name' => 'Bithumb', 'countries' => 'KR', // South Korea 'rateLimit' => 500, - 'hasCORS' => true, - // obsolete metainfo interface - 'hasFetchTickers' => true, - 'hasWithdraw' => true, - // new metainfo interface 'has' => array ( + 'CORS' => true, 'fetchTickers' => true, 'withdraw' => true, ), diff --git a/php/bitlish.php b/php/bitlish.php index d0300a53d980f..f5e85f909a839 100644 --- a/php/bitlish.php +++ b/php/bitlish.php @@ -11,10 +11,12 @@ public function describe () { 'countries' => array ( 'GB', 'EU', 'RU' ), 'rateLimit' => 1500, 'version' => 'v1', - 'hasCORS' => false, - 'hasFetchTickers' => true, - 'hasFetchOHLCV' => true, - 'hasWithdraw' => true, + 'has' => array ( + 'CORS' => false, + 'fetchTickers' => true, + 'fetchOHLCV' => true, + 'withdraw' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766275-dcfc6c30-5ed3-11e7-839d-00a846385d0b.jpg', 'api' => 'https://bitlish.com/api', diff --git a/php/bitmarket.php b/php/bitmarket.php index cc3a23fa7aaab..358c4a62d780e 100644 --- a/php/bitmarket.php +++ b/php/bitmarket.php @@ -10,9 +10,11 @@ public function describe () { 'name' => 'BitMarket', 'countries' => array ( 'PL', 'EU' ), 'rateLimit' => 1500, - 'hasCORS' => false, - 'hasFetchOHLCV' => true, - 'hasWithdraw' => true, + 'has' => array ( + 'CORS' => false, + 'fetchOHLCV' => true, + 'withdraw' => true, + ), 'timeframes' => array ( '90m' => '90m', '6h' => '6h', diff --git a/php/bitmex.php b/php/bitmex.php index cc3730fa6681a..b2e5e98492de0 100644 --- a/php/bitmex.php +++ b/php/bitmex.php @@ -12,9 +12,11 @@ public function describe () { 'version' => 'v1', 'userAgent' => null, 'rateLimit' => 1500, - 'hasCORS' => false, - 'hasFetchOHLCV' => true, - 'hasWithdraw' => true, + 'has' => array ( + 'CORS' => false, + 'fetchOHLCV' => true, + 'withdraw' => true, + ), 'timeframes' => array ( '1m' => '1m', '5m' => '5m', diff --git a/php/bitso.php b/php/bitso.php index 9d877431d329a..87ef632c5072e 100644 --- a/php/bitso.php +++ b/php/bitso.php @@ -11,7 +11,9 @@ public function describe () { 'countries' => 'MX', // Mexico 'rateLimit' => 2000, // 30 requests per minute 'version' => 'v3', - 'hasCORS' => true, + 'has' => array ( + 'CORS' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766335-715ce7aa-5ed5-11e7-88a8-173a27bb30fe.jpg', 'api' => 'https://api.bitso.com', diff --git a/php/bitstamp.php b/php/bitstamp.php index 92abb3b8064ba..346821e91d490 100644 --- a/php/bitstamp.php +++ b/php/bitstamp.php @@ -11,12 +11,8 @@ public function describe () { 'countries' => 'GB', 'rateLimit' => 1000, 'version' => 'v2', - 'hasCORS' => false, - // obsolete metainfo interface - 'hasFetchOrder' => true, - 'hasWithdraw' => true, - // new metainfo interface 'has' => array ( + 'CORS' => true, 'fetchOrder' => true, 'withdraw' => true, ), diff --git a/php/bitstamp1.php b/php/bitstamp1.php index 5c5e799e07463..2100774d3bd79 100644 --- a/php/bitstamp1.php +++ b/php/bitstamp1.php @@ -11,7 +11,9 @@ public function describe () { 'countries' => 'GB', 'rateLimit' => 1000, 'version' => 'v1', - 'hasCORS' => true, + 'has' => array ( + 'CORS' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27786377-8c8ab57e-5fe9-11e7-8ea4-2b05b6bcceec.jpg', 'api' => 'https://www.bitstamp.net/api', diff --git a/php/bittrex.php b/php/bittrex.php index f0d318230b40f..a3ccb63ab33c7 100644 --- a/php/bittrex.php +++ b/php/bittrex.php @@ -12,19 +12,9 @@ public function describe () { 'version' => 'v1.1', 'rateLimit' => 1500, 'hasAlreadyAuthenticatedSuccessfully' => false, // a workaround for APIKEY_INVALID - 'hasCORS' => false, - // obsolete metainfo interface - 'hasFetchTickers' => true, - 'hasFetchOHLCV' => true, - 'hasFetchOrder' => true, - 'hasFetchOrders' => true, - 'hasFetchClosedOrders' => true, - 'hasFetchOpenOrders' => true, - 'hasFetchMyTrades' => false, - 'hasFetchCurrencies' => true, - 'hasWithdraw' => true, // new metainfo interface 'has' => array ( + 'CORS' => true, 'fetchTickers' => true, 'fetchOHLCV' => true, 'fetchOrder' => true, diff --git a/php/bl3p.php b/php/bl3p.php index 06bdcadaefcc4..ba21bd2779e0c 100644 --- a/php/bl3p.php +++ b/php/bl3p.php @@ -12,7 +12,9 @@ public function describe () { 'rateLimit' => 1000, 'version' => '1', 'comment' => 'An exchange market by BitonicNL', - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/28501752-60c21b82-6feb-11e7-818b-055ee6d0e754.jpg', 'api' => 'https://api.bl3p.eu', diff --git a/php/bleutrade.php b/php/bleutrade.php index 21c8bdade1368..524be4386ac79 100644 --- a/php/bleutrade.php +++ b/php/bleutrade.php @@ -11,9 +11,11 @@ public function describe () { 'countries' => 'BR', // Brazil 'rateLimit' => 1000, 'version' => 'v2', - 'hasCORS' => true, - 'hasFetchTickers' => true, - 'hasFetchOHLCV' => false, + 'has' => array ( + 'CORS' => true, + 'fetchTickers' => true, + 'fetchOHLCV' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/30303000-b602dbe6-976d-11e7-956d-36c5049c01e7.jpg', 'api' => array ( diff --git a/php/braziliex.php b/php/braziliex.php index ce77c1cbe3530..d9622f351f9ee 100644 --- a/php/braziliex.php +++ b/php/braziliex.php @@ -10,11 +10,6 @@ public function describe () { 'name' => 'Braziliex', 'countries' => 'BR', 'rateLimit' => 1000, - // obsolete metainfo interface - 'hasFetchTickers' => true, - 'hasFetchOpenOrders' => true, - 'hasFetchMyTrades' => true, - // new metainfo interface 'has' => array ( 'fetchTickers' => true, 'fetchOpenOrders' => true, diff --git a/php/btcbox.php b/php/btcbox.php index 36f004fe9c2ba..2bd261e4b6964 100644 --- a/php/btcbox.php +++ b/php/btcbox.php @@ -11,8 +11,10 @@ public function describe () { 'countries' => 'JP', 'rateLimit' => 1000, 'version' => 'v1', - 'hasCORS' => false, - 'hasFetchOHLCV' => false, + 'has' => array ( + 'CORS' => false, + 'fetchOHLCV' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/31275803-4df755a8-aaa1-11e7-9abb-11ec2fad9f2d.jpg', 'api' => 'https://www.btcbox.co.jp/api', diff --git a/php/btcchina.php b/php/btcchina.php index 598ef96182e9f..cc20041f2f80d 100644 --- a/php/btcchina.php +++ b/php/btcchina.php @@ -11,7 +11,9 @@ public function describe () { 'countries' => 'CN', 'rateLimit' => 1500, 'version' => 'v1', - 'hasCORS' => true, + 'has' => array ( + 'CORS' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766368-465b3286-5ed6-11e7-9a11-0f6467e1d82b.jpg', 'api' => array ( diff --git a/php/btcexchange.php b/php/btcexchange.php index 3706889590e99..0accd595725ed 100644 --- a/php/btcexchange.php +++ b/php/btcexchange.php @@ -10,7 +10,9 @@ public function describe () { 'name' => 'BTCExchange', 'countries' => 'PH', // Philippines 'rateLimit' => 1500, - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27993052-4c92911a-64aa-11e7-96d8-ec6ac3435757.jpg', 'api' => 'https://www.btcexchange.ph/api', diff --git a/php/btcmarkets.php b/php/btcmarkets.php index 9bee8c7f53804..4b0352e1d3ecd 100644 --- a/php/btcmarkets.php +++ b/php/btcmarkets.php @@ -10,13 +10,8 @@ public function describe () { 'name' => 'BTC Markets', 'countries' => 'AU', // Australia 'rateLimit' => 1000, // market data cached for 1 second (trades cached for 2 seconds) - 'hasCORS' => false, - 'hasFetchOrder' => true, - 'hasFetchOrders' => true, - 'hasFetchClosedOrders' => true, - 'hasFetchOpenOrders' => true, - 'hasFetchMyTrades' => true, 'has' => array ( + 'CORS' => false, 'fetchOrder' => true, 'fetchOrders' => true, 'fetchClosedOrders' => 'emulated', diff --git a/php/btctradeua.php b/php/btctradeua.php index 6b9c123d61aad..09a564e96e7b9 100644 --- a/php/btctradeua.php +++ b/php/btctradeua.php @@ -10,7 +10,9 @@ public function describe () { 'name' => 'BTC Trade UA', 'countries' => 'UA', // Ukraine, 'rateLimit' => 3000, - 'hasCORS' => true, + 'has' => array ( + 'CORS' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27941483-79fc7350-62d9-11e7-9f61-ac47f28fcd96.jpg', 'api' => 'https://btc-trade.com.ua/api', diff --git a/php/btcturk.php b/php/btcturk.php index d8d9927743d33..9f601a2d87a19 100644 --- a/php/btcturk.php +++ b/php/btcturk.php @@ -10,9 +10,11 @@ public function describe () { 'name' => 'BTCTurk', 'countries' => 'TR', // Turkey 'rateLimit' => 1000, - 'hasCORS' => true, - 'hasFetchTickers' => true, - 'hasFetchOHLCV' => true, + 'has' => array ( + 'CORS' => true, + 'fetchTickers' => true, + 'fetchOHLCV' => true, + ), 'timeframes' => array ( '1d' => '1d', ), diff --git a/php/btcx.php b/php/btcx.php index 026dd50534043..581f735cf3881 100644 --- a/php/btcx.php +++ b/php/btcx.php @@ -11,7 +11,9 @@ public function describe () { 'countries' => array ( 'IS', 'US', 'EU' ), 'rateLimit' => 1500, // support in english is very poor, unable to tell rate limits 'version' => 'v1', - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766385-9fdcc98c-5ed6-11e7-8f14-66d5e5cd47e6.jpg', 'api' => 'https://btc-x.is/api', diff --git a/php/bter.php b/php/bter.php index 871f5a4862297..71ffab6ea1f26 100644 --- a/php/bter.php +++ b/php/bter.php @@ -10,11 +10,6 @@ public function describe () { 'name' => 'Bter', 'countries' => array ( 'VG', 'CN' ), // British Virgin Islands, China 'version' => '2', - // obsolete metainfo interface - 'hasCORS' => false, - 'hasFetchTickers' => true, - 'hasWithdraw' => true, - // new metainfo interface 'has' => array ( 'CORS' => false, 'fetchTickers' => true, diff --git a/php/bxinth.php b/php/bxinth.php index 6b294fba4a8e0..5b8cdf38b9f92 100644 --- a/php/bxinth.php +++ b/php/bxinth.php @@ -10,8 +10,10 @@ public function describe () { 'name' => 'BX.in.th', 'countries' => 'TH', // Thailand 'rateLimit' => 1500, - 'hasCORS' => false, - 'hasFetchTickers' => true, + 'has' => array ( + 'CORS' => false, + 'fetchTickers' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766412-567b1eb4-5ed7-11e7-94a8-ff6a3884f6c5.jpg', 'api' => 'https://bx.in.th/api', diff --git a/php/ccex.php b/php/ccex.php index 07beb628d3417..c53be2fb1ae89 100644 --- a/php/ccex.php +++ b/php/ccex.php @@ -10,8 +10,10 @@ public function describe () { 'name' => 'C-CEX', 'countries' => array ( 'DE', 'EU' ), 'rateLimit' => 1500, - 'hasCORS' => false, - 'hasFetchTickers' => true, + 'has' => array ( + 'CORS' => false, + 'fetchTickers' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766433-16881f90-5ed8-11e7-92f8-3d92cc747a6c.jpg', 'api' => array ( diff --git a/php/cex.php b/php/cex.php index 0f2dcb67fffef..4b40a7c0a640e 100644 --- a/php/cex.php +++ b/php/cex.php @@ -10,10 +10,12 @@ public function describe () { 'name' => 'CEX.IO', 'countries' => array ( 'GB', 'EU', 'CY', 'RU' ), 'rateLimit' => 1500, - 'hasCORS' => true, - 'hasFetchTickers' => true, - 'hasFetchOHLCV' => true, - 'hasFetchOpenOrders' => true, + 'has' => array ( + 'CORS' => true, + 'fetchTickers' => true, + 'fetchOHLCV' => true, + 'fetchOpenOrders' => true, + ), 'timeframes' => array ( '1m' => '1m', ), diff --git a/php/chbtc.php b/php/chbtc.php index a89eb7c047556..4e2be33d1d8de 100644 --- a/php/chbtc.php +++ b/php/chbtc.php @@ -11,8 +11,10 @@ public function describe () { 'countries' => 'CN', 'rateLimit' => 1000, 'version' => 'v1', - 'hasCORS' => false, - 'hasFetchOrder' => true, + 'has' => array ( + 'CORS' => false, + 'fetchOrder' => true + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/28555659-f0040dc2-7109-11e7-9d99-688a438bf9f4.jpg', 'api' => array ( diff --git a/php/chilebit.php b/php/chilebit.php index 7493c9151a5e5..8d673f3ef3a51 100644 --- a/php/chilebit.php +++ b/php/chilebit.php @@ -9,7 +9,9 @@ public function describe () { 'id' => 'chilebit', 'name' => 'ChileBit', 'countries' => 'CL', - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27991414-1298f0d8-647f-11e7-9c40-d56409266336.jpg', 'api' => array ( diff --git a/php/coincheck.php b/php/coincheck.php index 840e7c4e92446..fdc31e97777e8 100644 --- a/php/coincheck.php +++ b/php/coincheck.php @@ -10,7 +10,9 @@ public function describe () { 'name' => 'coincheck', 'countries' => array ( 'JP', 'ID' ), 'rateLimit' => 1500, - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766464-3b5c3c74-5ed9-11e7-840e-31b32968e1da.jpg', 'api' => 'https://coincheck.com/api', diff --git a/php/coinexchange.php b/php/coinexchange.php index ceebf673485ef..335b12a673556 100644 --- a/php/coinexchange.php +++ b/php/coinexchange.php @@ -10,13 +10,9 @@ public function describe () { 'name' => 'CoinExchange', 'countries' => array ( 'IN', 'JP', 'KR', 'VN', 'US' ), 'rateLimit' => 1000, - // obsolete metainfo interface - 'hasPrivateAPI' => false, - 'hasFetchTrades' => false, - 'hasFetchCurrencies' => true, - 'hasFetchTickers' => true, // new metainfo interface 'has' => array ( + 'privateAPI' => false, 'fetchTrades' => false, 'fetchCurrencies' => true, 'fetchTickers' => true, diff --git a/php/coinfloor.php b/php/coinfloor.php index 763713aaaedcf..e0a37ff09da40 100644 --- a/php/coinfloor.php +++ b/php/coinfloor.php @@ -10,7 +10,9 @@ public function describe () { 'name' => 'coinfloor', 'rateLimit' => 1000, 'countries' => 'UK', - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/28246081-623fc164-6a1c-11e7-913f-bac0d5576c90.jpg', 'api' => 'https://webapi.coinfloor.co.uk:8090/bist', diff --git a/php/coingi.php b/php/coingi.php index 6587f607e51f7..bc4d8ba620dcd 100644 --- a/php/coingi.php +++ b/php/coingi.php @@ -10,8 +10,10 @@ public function describe () { 'name' => 'Coingi', 'rateLimit' => 1000, 'countries' => array ( 'PA', 'BG', 'CN', 'US' ), // Panama, Bulgaria, China, US - 'hasFetchTickers' => true, - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + 'fetchTickers' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/28619707-5c9232a8-7212-11e7-86d6-98fe5d15cc6e.jpg', 'api' => array ( diff --git a/php/coinmarketcap.php b/php/coinmarketcap.php index 1211d40d510b5..76436fb6826ed 100644 --- a/php/coinmarketcap.php +++ b/php/coinmarketcap.php @@ -11,16 +11,16 @@ public function describe () { 'rateLimit' => 10000, 'version' => 'v1', 'countries' => 'US', - 'hasCORS' => true, - 'hasPrivateAPI' => false, - 'hasCreateOrder' => false, - 'hasCancelOrder' => false, - 'hasFetchBalance' => false, - 'hasFetchOrderBook' => false, - 'hasFetchTrades' => false, - 'hasFetchTickers' => true, - 'hasFetchCurrencies' => true, 'has' => array ( + 'CORS' => true, + 'privateAPI' => false, + 'createOrder' => false, + 'cancelOrder' => false, + 'fetchBalance' => false, + 'fetchOrderBook' => false, + 'fetchTrades' => false, + 'fetchTickers' => true, + 'fetchCurrencies' => true, 'fetchCurrencies' => true, ), 'urls' => array ( diff --git a/php/coinmate.php b/php/coinmate.php index a942bc9247415..e45d1e186f19d 100644 --- a/php/coinmate.php +++ b/php/coinmate.php @@ -10,7 +10,9 @@ public function describe () { 'name' => 'CoinMate', 'countries' => array ( 'GB', 'CZ', 'EU' ), // UK, Czech Republic 'rateLimit' => 1000, - 'hasCORS' => true, + 'has' => array ( + 'CORS' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27811229-c1efb510-606c-11e7-9a36-84ba2ce412d8.jpg', 'api' => 'https://coinmate.io/api', diff --git a/php/coinsecure.php b/php/coinsecure.php index f3768fc4fe7d4..4b2643c594b1b 100644 --- a/php/coinsecure.php +++ b/php/coinsecure.php @@ -11,7 +11,9 @@ public function describe () { 'countries' => 'IN', // India 'rateLimit' => 1000, 'version' => 'v1', - 'hasCORS' => true, + 'has' => array ( + 'CORS' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766472-9cbd200a-5ed9-11e7-9551-2267ad7bac08.jpg', 'api' => 'https://api.coinsecure.in', diff --git a/php/coinspot.php b/php/coinspot.php index 344bf5dc0f517..e69aa87d74dbc 100644 --- a/php/coinspot.php +++ b/php/coinspot.php @@ -10,7 +10,9 @@ public function describe () { 'name' => 'CoinSpot', 'countries' => 'AU', // Australia 'rateLimit' => 1000, - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/28208429-3cacdf9a-6896-11e7-854e-4c79a772a30f.jpg', 'api' => array ( diff --git a/php/cryptopia.php b/php/cryptopia.php index 08188692805f4..a0afe2d3dca6a 100644 --- a/php/cryptopia.php +++ b/php/cryptopia.php @@ -10,19 +10,8 @@ public function describe () { 'name' => 'Cryptopia', 'rateLimit' => 1500, 'countries' => 'NZ', // New Zealand - 'hasCORS' => false, - // obsolete metainfo interface - 'hasFetchTickers' => true, - 'hasFetchOrder' => true, - 'hasFetchOrders' => true, - 'hasFetchOpenOrders' => true, - 'hasFetchClosedOrders' => true, - 'hasFetchMyTrades' => true, - 'hasFetchCurrencies' => true, - 'hasDeposit' => true, - 'hasWithdraw' => true, - // new metainfo interface 'has' => array ( + 'CORS' => false, 'fetchTickers' => true, 'fetchOrder' => 'emulated', 'fetchOrders' => 'emulated', diff --git a/php/dsx.php b/php/dsx.php index 228b6ea783f01..f7f95ed4bdf2d 100644 --- a/php/dsx.php +++ b/php/dsx.php @@ -10,13 +10,15 @@ public function describe () { 'name' => 'DSX', 'countries' => 'UK', 'rateLimit' => 1500, - 'hasCORS' => false, - 'hasFetchOrder' => true, - 'hasFetchOrders' => true, - 'hasFetchOpenOrders' => true, - 'hasFetchClosedOrders' => true, - 'hasFetchTickers' => true, - 'hasFetchMyTrades' => true, + 'has' => array ( + 'CORS' => false, + 'fetchOrder' => true, + 'fetchOrders' => true, + 'fetchOpenOrders' => true, + 'fetchClosedOrders' => true, + 'fetchTickers' => true, + 'fetchMyTrades' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27990275-1413158a-645a-11e7-931c-94717f7510e3.jpg', 'api' => array ( diff --git a/php/exmo.php b/php/exmo.php index bf536b148c30c..3fc99a6f3b7a1 100644 --- a/php/exmo.php +++ b/php/exmo.php @@ -11,9 +11,11 @@ public function describe () { 'countries' => array ( 'ES', 'RU' ), // Spain, Russia 'rateLimit' => 1000, // once every 350 ms ≈ 180 requests per minute ≈ 3 requests per second 'version' => 'v1', - 'hasCORS' => false, - 'hasFetchTickers' => true, - 'hasWithdraw' => true, + 'has' => array ( + 'CORS' => false, + 'fetchTickers' => true, + 'withdraw' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766491-1b0ea956-5eda-11e7-9225-40d67b481b8d.jpg', 'api' => 'https://api.exmo.com', diff --git a/php/flowbtc.php b/php/flowbtc.php index 81cd54541fd48..4f3f819909b29 100644 --- a/php/flowbtc.php +++ b/php/flowbtc.php @@ -11,7 +11,9 @@ public function describe () { 'countries' => 'BR', // Brazil 'version' => 'v1', 'rateLimit' => 1000, - 'hasCORS' => true, + 'has' => array ( + 'CORS' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/28162465-cd815d4c-67cf-11e7-8e57-438bea0523a2.jpg', 'api' => 'https://api.flowbtc.com:8400/ajax', diff --git a/php/foxbit.php b/php/foxbit.php index e8ed754e72a12..918ab80338760 100644 --- a/php/foxbit.php +++ b/php/foxbit.php @@ -9,7 +9,9 @@ public function describe () { 'id' => 'foxbit', 'name' => 'FoxBit', 'countries' => 'BR', - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'rateLimit' => 1000, 'version' => 'v1', 'urls' => array ( diff --git a/php/fybse.php b/php/fybse.php index 58423470035ff..6f151780f7ab7 100644 --- a/php/fybse.php +++ b/php/fybse.php @@ -9,7 +9,9 @@ public function describe () { 'id' => 'fybse', 'name' => 'FYB-SE', 'countries' => 'SE', // Sweden - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'rateLimit' => 1500, 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766512-31019772-5edb-11e7-8241-2e675e6797f1.jpg', diff --git a/php/fybsg.php b/php/fybsg.php index fc22388d5d6cb..2d2c838534ddf 100644 --- a/php/fybsg.php +++ b/php/fybsg.php @@ -9,7 +9,9 @@ public function describe () { 'id' => 'fybsg', 'name' => 'FYB-SG', 'countries' => 'SG', // Singapore - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766513-3364d56a-5edb-11e7-9e6b-d5898bb89c81.jpg', 'api' => 'https://www.fybsg.com/api/SGD', diff --git a/php/gatecoin.php b/php/gatecoin.php index ae0bd20e2110a..a77bdcc582a71 100644 --- a/php/gatecoin.php +++ b/php/gatecoin.php @@ -11,9 +11,11 @@ public function describe () { 'rateLimit' => 2000, 'countries' => 'HK', // Hong Kong 'comment' => 'a regulated/licensed exchange', - 'hasCORS' => false, - 'hasFetchTickers' => true, - 'hasFetchOHLCV' => true, + 'has' => array ( + 'CORS' => false, + 'fetchTickers' => true, + 'fetchOHLCV' => true, + ), 'timeframes' => array ( '1m' => '1m', '15m' => '15m', diff --git a/php/gdax.php b/php/gdax.php index caf350491a0c2..c970f92c04a14 100644 --- a/php/gdax.php +++ b/php/gdax.php @@ -11,16 +11,6 @@ public function describe () { 'countries' => 'US', 'rateLimit' => 1000, 'userAgent' => $this->userAgents['chrome'], - // obsolete metainfo interface - 'hasCORS' => true, - 'hasFetchOHLCV' => true, - 'hasDeposit' => true, - 'hasWithdraw' => true, - 'hasFetchOrder' => true, - 'hasFetchOrders' => true, - 'hasFetchOpenOrders' => true, - 'hasFetchClosedOrders' => true, - // new metainfo interface 'has' => array ( 'CORS' => true, 'fetchOHLCV' => true, diff --git a/php/gemini.php b/php/gemini.php index 450cfeb202037..17188f02b5f6e 100644 --- a/php/gemini.php +++ b/php/gemini.php @@ -11,10 +11,6 @@ public function describe () { 'countries' => 'US', 'rateLimit' => 1500, // 200 for private API 'version' => 'v1', - // obsolete metainfo interface - 'hasCORS' => false, - 'hasWithdraw' => true, - // new metainfo interface 'has' => array ( 'CORS' => false, 'withdraw' => true, diff --git a/php/hitbtc.php b/php/hitbtc.php index deca598a4cf66..76793adc5ea78 100644 --- a/php/hitbtc.php +++ b/php/hitbtc.php @@ -11,12 +11,13 @@ public function describe () { 'countries' => 'UK', 'rateLimit' => 1500, 'version' => '1', - 'hasCORS' => false, - 'hasFetchTickers' => true, - 'hasFetchOrder' => true, - 'hasFetchOpenOrders' => true, - 'hasFetchClosedOrders' => true, - 'hasWithdraw' => true, + 'has' => array ( + 'CORS' => false, + 'fetchOrder' => true, + 'fetchOpenOrders' => true, + 'fetchClosedOrders' => true, + 'withdraw' => true + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766555-8eaec20e-5edc-11e7-9c5b-6dc69fc42f5e.jpg', 'api' => 'http://api.hitbtc.com', diff --git a/php/hitbtc2.php b/php/hitbtc2.php index fa3c51aee2602..f1a7c6f29e5ed 100644 --- a/php/hitbtc2.php +++ b/php/hitbtc2.php @@ -11,19 +11,8 @@ public function describe () { 'countries' => 'UK', 'rateLimit' => 1500, 'version' => '2', - 'hasCORS' => true, - // older metainfo interface - 'hasFetchOHLCV' => true, - 'hasFetchTickers' => true, - 'hasFetchOrder' => true, - 'hasFetchOrders' => false, - 'hasFetchOpenOrders' => true, - 'hasFetchClosedOrders' => true, - 'hasFetchMyTrades' => true, - 'hasWithdraw' => true, - 'hasFetchCurrencies' => true, - // new metainfo interface 'has' => array ( + 'CORS' => true, 'fetchCurrencies' => true, 'fetchOHLCV' => true, 'fetchTickers' => true, diff --git a/php/huobi.php b/php/huobi.php index 881e574280de9..e877362e6e9fa 100644 --- a/php/huobi.php +++ b/php/huobi.php @@ -11,8 +11,10 @@ public function describe () { 'countries' => 'CN', 'rateLimit' => 2000, 'version' => 'v3', - 'hasCORS' => false, - 'hasFetchOHLCV' => true, + 'has' => array ( + 'CORS' => false, + 'fetchOHLCV' => true, + ), 'timeframes' => array ( '1m' => '001', '5m' => '005', diff --git a/php/huobicny.php b/php/huobicny.php index e31899d8d2984..b755fed6a2a0b 100644 --- a/php/huobicny.php +++ b/php/huobicny.php @@ -9,7 +9,9 @@ public function describe () { 'id' => 'huobicny', 'name' => 'Huobi CNY', 'hostname' => 'be.huobi.com', - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766569-15aa7b9a-5edd-11e7-9e7f-44791f4ee49c.jpg', 'api' => 'https://be.huobi.com', diff --git a/php/huobipro.php b/php/huobipro.php index 4de99d1a6df3d..76a2cf241dd24 100644 --- a/php/huobipro.php +++ b/php/huobipro.php @@ -15,13 +15,8 @@ public function describe () { 'accounts' => null, 'accountsById' => null, 'hostname' => 'api.huobi.pro', - 'hasCORS' => false, - // obsolete metainfo structure - 'hasFetchOHLCV' => true, - 'hasFetchOrders' => true, - 'hasFetchOpenOrders' => true, - // new metainfo structure 'has' => array ( + 'CORS' => false, 'fetchOHCLV' => true, 'fetchOrders' => true, 'fetchOpenOrders' => true, diff --git a/php/independentreserve.php b/php/independentreserve.php index 9afbb81d28a2a..e18f407ab1b8b 100644 --- a/php/independentreserve.php +++ b/php/independentreserve.php @@ -10,7 +10,9 @@ public function describe () { 'name' => 'Independent Reserve', 'countries' => array ( 'AU', 'NZ' ), // Australia, New Zealand 'rateLimit' => 1000, - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/30521662-cf3f477c-9bcb-11e7-89bc-d1ac85012eda.jpg', 'api' => array ( diff --git a/php/itbit.php b/php/itbit.php index fe50c5d86aae2..1c0c31be1acc5 100644 --- a/php/itbit.php +++ b/php/itbit.php @@ -11,7 +11,9 @@ public function describe () { 'countries' => 'US', 'rateLimit' => 2000, 'version' => 'v1', - 'hasCORS' => true, + 'has' => array ( + 'CORS' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27822159-66153620-60ad-11e7-89e7-005f6d7f3de0.jpg', 'api' => 'https://api.itbit.com', diff --git a/php/jubi.php b/php/jubi.php index cec62cfb1d7b2..54dd6a98564a1 100644 --- a/php/jubi.php +++ b/php/jubi.php @@ -11,8 +11,10 @@ public function describe () { 'countries' => 'CN', 'rateLimit' => 1500, 'version' => 'v1', - 'hasCORS' => false, - 'hasFetchTickers' => true, + 'has' => array ( + 'CORS' => false, + 'fetchTickers' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766581-9d397d9a-5edd-11e7-8fb9-5d8236c0e692.jpg', 'api' => 'https://www.jubi.com/api', diff --git a/php/kraken.php b/php/kraken.php index 1aa2946ebb512..fad6448efe141 100644 --- a/php/kraken.php +++ b/php/kraken.php @@ -11,18 +11,8 @@ public function describe () { 'countries' => 'US', 'version' => '0', 'rateLimit' => 3000, - 'hasCORS' => false, - // obsolete metainfo interface - 'hasFetchTickers' => true, - 'hasFetchOHLCV' => true, - 'hasFetchOrder' => true, - 'hasFetchOpenOrders' => true, - 'hasFetchClosedOrders' => true, - 'hasFetchMyTrades' => true, - 'hasWithdraw' => true, - 'hasFetchCurrencies' => true, - // new metainfo interface 'has' => array ( + 'CORS' => false, 'fetchCurrencies' => true, 'fetchTickers' => true, 'fetchOHLCV' => true, diff --git a/php/kucoin.php b/php/kucoin.php index 1c22f81e22e59..3e7a784ddb4fc 100644 --- a/php/kucoin.php +++ b/php/kucoin.php @@ -11,20 +11,9 @@ public function describe () { 'countries' => 'HK', // Hong Kong 'version' => 'v1', 'rateLimit' => 2000, - 'hasCORS' => false, 'userAgent' => $this->userAgents['chrome'], - // obsolete metainfo interface - 'hasFetchTickers' => true, - 'hasFetchOHLCV' => true, - 'hasFetchOrder' => false, - 'hasFetchOrders' => true, - 'hasFetchClosedOrders' => true, - 'hasFetchOpenOrders' => true, - 'hasFetchMyTrades' => false, - 'hasFetchCurrencies' => true, - 'hasWithdraw' => true, - // new metainfo interface 'has' => array ( + 'CORS' => false, 'fetchTickers' => true, 'fetchOHLCV' => true, // see the method implementation below 'fetchOrder' => false, diff --git a/php/kuna.php b/php/kuna.php index d1d76561c6775..2dfc0581a56cf 100644 --- a/php/kuna.php +++ b/php/kuna.php @@ -11,9 +11,11 @@ public function describe () { 'countries' => 'UA', 'rateLimit' => 1000, 'version' => 'v2', - 'hasCORS' => false, - 'hasFetchTickers' => false, - 'hasFetchOHLCV' => false, + 'has' => array ( + 'CORS' => false, + 'fetchTickers' => false, + 'fetchOHLCV' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/31697638-912824fa-b3c1-11e7-8c36-cf9606eb94ac.jpg', 'api' => 'https://kuna.io', diff --git a/php/lakebtc.php b/php/lakebtc.php index b65d5a0fd3827..f99be51e6cf74 100644 --- a/php/lakebtc.php +++ b/php/lakebtc.php @@ -10,7 +10,9 @@ public function describe () { 'name' => 'LakeBTC', 'countries' => 'US', 'version' => 'api_v2', - 'hasCORS' => true, + 'has' => array ( + 'CORS' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/28074120-72b7c38a-6660-11e7-92d9-d9027502281d.jpg', 'api' => 'https://api.lakebtc.com', diff --git a/php/liqui.php b/php/liqui.php index 871421d666903..18bb1061e76aa 100644 --- a/php/liqui.php +++ b/php/liqui.php @@ -11,18 +11,9 @@ public function describe () { 'countries' => 'UA', 'rateLimit' => 3000, 'version' => '3', - 'hasCORS' => false, 'userAgent' => $this->userAgents['chrome'], - // obsolete metainfo interface - 'hasFetchOrder' => true, - 'hasFetchOrders' => true, - 'hasFetchOpenOrders' => true, - 'hasFetchClosedOrders' => true, - 'hasFetchTickers' => true, - 'hasFetchMyTrades' => true, - 'hasWithdraw' => true, - // new metainfo interface 'has' => array ( + 'CORS' => false, 'fetchOrder' => true, 'fetchOrders' => 'emulated', 'fetchOpenOrders' => true, diff --git a/php/livecoin.php b/php/livecoin.php index 592e60d1426f1..55279a88f34ac 100644 --- a/php/livecoin.php +++ b/php/livecoin.php @@ -10,12 +10,8 @@ public function describe () { 'name' => 'LiveCoin', 'countries' => array ( 'US', 'UK', 'RU' ), 'rateLimit' => 1000, - 'hasCORS' => false, - // obsolete metainfo interface - 'hasFetchTickers' => true, - 'hasFetchCurrencies' => true, - // new metainfo interface 'has' => array ( + 'CORS' => false, 'fetchTickers' => true, 'fetchCurrencies' => true, ), diff --git a/php/luno.php b/php/luno.php index b1961dfb01352..7e341402bb1a8 100644 --- a/php/luno.php +++ b/php/luno.php @@ -11,10 +11,8 @@ public function describe () { 'countries' => array ( 'GB', 'SG', 'ZA' ), 'rateLimit' => 10000, 'version' => '1', - 'hasCORS' => false, - 'hasFetchTickers' => true, - 'hasFetchOrder' => true, 'has' => array ( + 'CORS' => false, 'fetchTickers' => true, 'fetchOrder' => true, ), diff --git a/php/lykke.php b/php/lykke.php index 87a2285f18798..2cc1e3d0d4d60 100644 --- a/php/lykke.php +++ b/php/lykke.php @@ -11,12 +11,8 @@ public function describe () { 'countries' => 'CH', 'version' => 'v1', 'rateLimit' => 200, - 'hasCORS' => false, - // obsolete metainfo interface - 'hasFetchTrades' => false, - 'hasFetchOHLCV' => false, - // new metainfo interface 'has' => array ( + 'CORS' => false, 'fetchOHLCV' => false, 'fetchTrades' => false, ), diff --git a/php/mercado.php b/php/mercado.php index 33f3c5142de2c..bc7c5845e5f02 100644 --- a/php/mercado.php +++ b/php/mercado.php @@ -11,8 +11,10 @@ public function describe () { 'countries' => 'BR', // Brazil 'rateLimit' => 1000, 'version' => 'v3', - 'hasCORS' => true, - 'hasWithdraw' => true, + 'has' => array ( + 'CORS' => true, + 'withdraw' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27837060-e7c58714-60ea-11e7-9192-f05e86adb83f.jpg', 'api' => array ( diff --git a/php/mixcoins.php b/php/mixcoins.php index 93d3177d3905e..4e3f5321aae26 100644 --- a/php/mixcoins.php +++ b/php/mixcoins.php @@ -11,7 +11,9 @@ public function describe () { 'countries' => array ( 'GB', 'HK' ), 'rateLimit' => 1500, 'version' => 'v1', - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/30237212-ed29303c-9535-11e7-8af8-fcd381cfa20c.jpg', 'api' => 'https://mixcoins.com/api', diff --git a/php/nova.php b/php/nova.php index f28ee567ffba6..96dc6eee178ad 100644 --- a/php/nova.php +++ b/php/nova.php @@ -11,7 +11,9 @@ public function describe () { 'countries' => 'TZ', // Tanzania 'rateLimit' => 2000, 'version' => 'v2', - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/30518571-78ca0bca-9b8a-11e7-8840-64b83a4a94b2.jpg', 'api' => 'https://novaexchange.com/remote', diff --git a/php/okcoincny.php b/php/okcoincny.php index 8a7590362372f..5334dac083d49 100644 --- a/php/okcoincny.php +++ b/php/okcoincny.php @@ -9,7 +9,9 @@ public function describe () { 'id' => 'okcoincny', 'name' => 'OKCoin CNY', 'countries' => 'CN', - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766792-8be9157a-5ee5-11e7-926c-6d69b8d3378d.jpg', 'api' => array ( diff --git a/php/okcoinusd.php b/php/okcoinusd.php index 8c1b4b8b9bd92..2cdd3b0bb74a7 100644 --- a/php/okcoinusd.php +++ b/php/okcoinusd.php @@ -9,27 +9,19 @@ public function describe () { 'id' => 'okcoinusd', 'name' => 'OKCoin USD', 'countries' => array ( 'CN', 'US' ), - 'hasCORS' => false, 'version' => 'v1', 'rateLimit' => 1000, // up to 3000 requests per 5 minutes ≈ 600 requests per minute ≈ 10 requests per second ≈ 100 ms - // obsolete metainfo interface - 'hasFetchOHLCV' => true, - 'hasFetchOrder' => true, - 'hasFetchOrders' => false, - 'hasFetchOpenOrders' => true, - 'hasFetchClosedOrders' => true, - 'hasWithdraw' => true, - // new metainfo interface 'has' => array ( + 'CORS' => false, 'fetchOHLCV' => true, 'fetchOrder' => true, 'fetchOrders' => false, 'fetchOpenOrders' => true, 'fetchClosedOrders' => true, 'withdraw' => true, + 'futureMarkets' => false, ), 'extension' => '.do', // appended to endpoint URL - 'hasFutureMarkets' => false, 'timeframes' => array ( '1m' => '1min', '3m' => '3min', @@ -187,7 +179,7 @@ public function fetch_markets () { ), )); $result[] = $market; - if (($this->hasFutureMarkets) && ($market['quote'] === 'USDT')) { + if (($this->has['futureMarkets']) && ($market['quote'] === 'USDT')) { $result[] = array_merge ($market, array ( 'quote' => 'USD', 'symbol' => $market['base'] . '/USD', diff --git a/php/okex.php b/php/okex.php index 63589862c346d..ca439a637d82c 100644 --- a/php/okex.php +++ b/php/okex.php @@ -9,8 +9,10 @@ public function describe () { 'id' => 'okex', 'name' => 'OKEX', 'countries' => array ( 'CN', 'US' ), - 'hasCORS' => false, - 'hasFutureMarkets' => true, + 'has' => array ( + 'CORS' => false, + 'futureMarkets' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/32552768-0d6dd3c6-c4a6-11e7-90f8-c043b64756a7.jpg', 'api' => array ( diff --git a/php/paymium.php b/php/paymium.php index 04963704db7f5..6d08188318fe5 100644 --- a/php/paymium.php +++ b/php/paymium.php @@ -11,7 +11,9 @@ public function describe () { 'countries' => array ( 'FR', 'EU' ), 'rateLimit' => 2000, 'version' => 'v1', - 'hasCORS' => true, + 'has' => array ( + 'CORS' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27790564-a945a9d4-5ff9-11e7-9d2d-b635763f2f24.jpg', 'api' => 'https://paymium.com/api', diff --git a/php/poloniex.php b/php/poloniex.php index 4ebd1d843f380..f26572d9ad758 100644 --- a/php/poloniex.php +++ b/php/poloniex.php @@ -10,19 +10,8 @@ public function describe () { 'name' => 'Poloniex', 'countries' => 'US', 'rateLimit' => 1000, // up to 6 calls per second - 'hasCORS' => true, - // obsolete metainfo interface - 'hasFetchMyTrades' => true, - 'hasFetchOrder' => true, - 'hasFetchOrders' => true, - 'hasFetchOpenOrders' => true, - 'hasFetchClosedOrders' => true, - 'hasFetchTickers' => true, - 'hasFetchCurrencies' => true, - 'hasWithdraw' => true, - 'hasFetchOHLCV' => true, - // new metainfo interface 'has' => array ( + 'CORS' => true, 'fetchOHLCV' => true, 'fetchMyTrades' => true, 'fetchOrder' => 'emulated', diff --git a/php/qryptos.php b/php/qryptos.php index 9dfb0274ecc10..37583305f95fb 100644 --- a/php/qryptos.php +++ b/php/qryptos.php @@ -11,9 +11,9 @@ public function describe () { 'countries' => array ( 'CN', 'TW' ), 'version' => '2', 'rateLimit' => 1000, - 'hasFetchTickers' => true, - 'hasCORS' => false, 'has' => array ( + 'CORS' => false, + 'fetchTickers' => true, 'fetchOrder' => true, 'fetchOrders' => true, 'fetchOpenOrders' => true, diff --git a/php/quadrigacx.php b/php/quadrigacx.php index ebde36e84b8b8..8f287627c387d 100644 --- a/php/quadrigacx.php +++ b/php/quadrigacx.php @@ -11,11 +11,8 @@ public function describe () { 'countries' => 'CA', 'rateLimit' => 1000, 'version' => 'v2', - 'hasCORS' => true, - // obsolete metainfo interface - 'hasWithdraw' => true, - // new metainfo interface 'has' => array ( + 'CORS' => true, 'withdraw' => true, ), 'urls' => array ( diff --git a/php/quoinex.php b/php/quoinex.php index 38d7a8d3939ae..e4abd28906269 100644 --- a/php/quoinex.php +++ b/php/quoinex.php @@ -11,8 +11,10 @@ public function describe () { 'countries' => array ( 'JP', 'SG', 'VN' ), 'version' => '2', 'rateLimit' => 1000, - 'hasFetchTickers' => true, - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + 'fetchTickers' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/35047114-0e24ad4a-fbaa-11e7-96a9-69c1a756083b.jpg', 'api' => 'https://api.quoine.com', diff --git a/php/southxchange.php b/php/southxchange.php index e1acb2667728e..0aab2d95ddf4a 100644 --- a/php/southxchange.php +++ b/php/southxchange.php @@ -10,9 +10,11 @@ public function describe () { 'name' => 'SouthXchange', 'countries' => 'AR', // Argentina 'rateLimit' => 1000, - 'hasFetchTickers' => true, - 'hasCORS' => false, - 'hasWithdraw' => true, + 'has' => array ( + 'CORS' => true, + 'fetchTickers' => true, + 'withdraw' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27838912-4f94ec8a-60f6-11e7-9e5d-bbf9bd50a559.jpg', 'api' => 'https://www.southxchange.com/api', diff --git a/php/surbitcoin.php b/php/surbitcoin.php index 932accd091a00..4747a9c962378 100644 --- a/php/surbitcoin.php +++ b/php/surbitcoin.php @@ -9,7 +9,9 @@ public function describe () { 'id' => 'surbitcoin', 'name' => 'SurBitcoin', 'countries' => 'VE', - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27991511-f0a50194-6481-11e7-99b5-8f02932424cc.jpg', 'api' => array ( diff --git a/php/test/test.php b/php/test/test.php index 15d04ac4c7e95..f1866e8a43858 100644 --- a/php/test/test.php +++ b/php/test/test.php @@ -79,7 +79,7 @@ function test_order_book ($exchange, $symbol) { function test_trades ($exchange, $symbol) { - if ($exchange->hasFetchTrades) { + if ($exchange->has['fetchTrades']) { $delay = $exchange->rateLimit * 1000; usleep ($delay); diff --git a/php/therock.php b/php/therock.php index 862445092e0a5..63edf28215d55 100644 --- a/php/therock.php +++ b/php/therock.php @@ -11,8 +11,10 @@ public function describe () { 'countries' => 'MT', 'rateLimit' => 1000, 'version' => 'v1', - 'hasCORS' => false, - 'hasFetchTickers' => true, + 'has' => array ( + 'CORS' => false, + 'fetchTickers' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766869-75057fa2-5ee9-11e7-9a6f-13e641fa4707.jpg', 'api' => 'https://api.therocktrading.com', diff --git a/php/tidex.php b/php/tidex.php index 972af1ee81ba0..cb9ebadff39a4 100644 --- a/php/tidex.php +++ b/php/tidex.php @@ -11,8 +11,10 @@ public function describe () { 'countries' => 'UK', 'rateLimit' => 2000, 'version' => '3', - // 'hasCORS' => false, - // 'hasFetchTickers' => true, + 'has' => array ( + // 'CORS' => false, + // 'fetchTickers' => true + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/30781780-03149dc4-a12e-11e7-82bb-313b269d24d4.jpg', 'api' => array ( diff --git a/php/urdubit.php b/php/urdubit.php index 92f2ea94cef66..96ffee2826547 100644 --- a/php/urdubit.php +++ b/php/urdubit.php @@ -9,7 +9,9 @@ public function describe () { 'id' => 'urdubit', 'name' => 'UrduBit', 'countries' => 'PK', - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27991453-156bf3ae-6480-11e7-82eb-7295fe1b5bb4.jpg', 'api' => array ( diff --git a/php/vaultoro.php b/php/vaultoro.php index 4b1a2c11db5ac..6b684d960e8d5 100644 --- a/php/vaultoro.php +++ b/php/vaultoro.php @@ -11,7 +11,9 @@ public function describe () { 'countries' => 'CH', 'rateLimit' => 1000, 'version' => '1', - 'hasCORS' => true, + 'has' => array ( + 'CORS' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766880-f205e870-5ee9-11e7-8fe2-0d5b15880752.jpg', 'api' => 'https://api.vaultoro.com', diff --git a/php/vbtc.php b/php/vbtc.php index 5577f76d6a58f..636863293fb08 100644 --- a/php/vbtc.php +++ b/php/vbtc.php @@ -9,7 +9,9 @@ public function describe () { 'id' => 'vbtc', 'name' => 'VBTC', 'countries' => 'VN', - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27991481-1f53d1d8-6481-11e7-884e-21d17e7939db.jpg', 'api' => array ( diff --git a/php/virwox.php b/php/virwox.php index 01b031adbe8e6..5b4aafe5a1165 100644 --- a/php/virwox.php +++ b/php/virwox.php @@ -10,7 +10,9 @@ public function describe () { 'name' => 'VirWoX', 'countries' => array ( 'AT', 'EU' ), 'rateLimit' => 1000, - 'hasCORS' => true, + 'has' => array ( + 'CORS' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766894-6da9d360-5eea-11e7-90aa-41f2711b7405.jpg', 'api' => array ( diff --git a/php/wex.php b/php/wex.php index 70b5d64f1488f..2162d22a695f4 100644 --- a/php/wex.php +++ b/php/wex.php @@ -10,8 +10,10 @@ public function describe () { 'name' => 'WEX', 'countries' => 'NZ', // New Zealand 'version' => '3', - 'hasFetchTickers' => true, - 'hasCORS' => false, + 'has' => array ( + 'CORS' => false, + 'fetchTickers' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/30652751-d74ec8f8-9e31-11e7-98c5-71469fcef03e.jpg', 'api' => array ( diff --git a/php/xbtce.php b/php/xbtce.php index 871064b0c40a9..9cfd51cb39bcd 100644 --- a/php/xbtce.php +++ b/php/xbtce.php @@ -11,10 +11,12 @@ public function describe () { 'countries' => 'RU', 'rateLimit' => 2000, // responses are cached every 2 seconds 'version' => 'v1', - 'hasPublicAPI' => false, - 'hasCORS' => false, - 'hasFetchTickers' => true, - 'hasFetchOHLCV' => false, + 'has' => array ( + 'publicAPI' => false, + 'CORS' => false, + 'fetchTickers' => true, + 'fetchOHLCV' => false, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/28059414-e235970c-662c-11e7-8c3a-08e31f78684b.jpg', 'api' => 'https://cryptottlivewebapi.xbtce.net:8443/api', diff --git a/php/yobit.php b/php/yobit.php index 4a2c2cef0be8d..7d6aadefe9c19 100644 --- a/php/yobit.php +++ b/php/yobit.php @@ -11,8 +11,10 @@ public function describe () { 'countries' => 'RU', 'rateLimit' => 3000, // responses are cached every 2 seconds 'version' => '3', - 'hasCORS' => false, - 'hasWithdraw' => true, + 'has' => array ( + 'CORS' => false, + 'withdraw' => true + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766910-cdcbfdae-5eea-11e7-9859-03fea873272d.jpg', 'api' => array ( diff --git a/php/yunbi.php b/php/yunbi.php index fe4561eeda036..bbc0ca3308a32 100644 --- a/php/yunbi.php +++ b/php/yunbi.php @@ -11,9 +11,11 @@ public function describe () { 'countries' => 'CN', 'rateLimit' => 1000, 'version' => 'v2', - 'hasCORS' => false, - 'hasFetchTickers' => true, - 'hasFetchOHLCV' => true, + 'has' => array ( + 'CORS' => false, + 'fetchTickers' => true, + 'fetchOHLCV' => true, + ), 'timeframes' => array ( '1m' => '1', '5m' => '5', diff --git a/php/zaif.php b/php/zaif.php index 8cf58466d1ffc..d107fe3820e56 100644 --- a/php/zaif.php +++ b/php/zaif.php @@ -11,10 +11,12 @@ public function describe () { 'countries' => 'JP', 'rateLimit' => 2000, 'version' => '1', - 'hasCORS' => false, - 'hasFetchOpenOrders' => true, - 'hasFetchClosedOrders' => true, - 'hasWithdraw' => true, + 'has' => array ( + 'CORS' => false, + 'fetchOpenOrders' => true, + 'fetchClosedOrders' => true, + 'withdraw' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/27766927-39ca2ada-5eeb-11e7-972f-1b4199518ca6.jpg', 'api' => 'https://api.zaif.jp', diff --git a/php/zb.php b/php/zb.php index 072027a93af4c..c33bcb0a1d3fe 100644 --- a/php/zb.php +++ b/php/zb.php @@ -11,8 +11,10 @@ public function describe () { 'countries' => 'CN', 'rateLimit' => 1000, 'version' => 'v1', - 'hasCORS' => false, - 'hasFetchOrder' => true, + 'has' => array ( + 'CORS' => false, + 'fetchOrder' => true, + ), 'urls' => array ( 'logo' => 'https://user-images.githubusercontent.com/1294454/32859187-cd5214f0-ca5e-11e7-967d-96568e2e2bd1.jpg', 'api' => array ( diff --git a/python/ccxt/_1broker.py b/python/ccxt/_1broker.py index f1168727a8cd2..5cc8f8c84c82b 100644 --- a/python/ccxt/_1broker.py +++ b/python/ccxt/_1broker.py @@ -13,10 +13,12 @@ def describe(self): 'countries': 'US', 'rateLimit': 1500, 'version': 'v2', - 'hasPublicAPI': False, - 'hasCORS': True, - 'hasFetchTrades': False, - 'hasFetchOHLCV': True, + 'has': { + 'publicAPI': False, + 'CORS': True, + 'fetchTrades': False, + 'fetchOHLCV': True, + }, 'timeframes': { '1m': '60', '15m': '900', diff --git a/python/ccxt/_1btcxe.py b/python/ccxt/_1btcxe.py index 865b7a478758c..5b07dc3d5631b 100644 --- a/python/ccxt/_1btcxe.py +++ b/python/ccxt/_1btcxe.py @@ -12,9 +12,11 @@ def describe(self): 'name': '1BTCXE', 'countries': 'PA', # Panama 'comment': 'Crypto Capital API', - 'hasCORS': True, - 'hasFetchOHLCV': True, - 'hasWithdraw': True, + 'has': { + 'CORS': True, + 'fetchTickers': True, + 'withdraw': True, + }, 'timeframes': { '1d': '1year', }, diff --git a/python/ccxt/acx.py b/python/ccxt/acx.py index 6d5d780baa6f3..d1d449f92b3d6 100644 --- a/python/ccxt/acx.py +++ b/python/ccxt/acx.py @@ -14,10 +14,12 @@ def describe(self): 'countries': 'AU', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': True, - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, - 'hasWithdraw': True, + 'has': { + 'CORS': True, + 'fetchTickers': True, + 'fetchOHLCV': True, + 'withdraw': True, + }, 'timeframes': { '1m': '1', '5m': '5', diff --git a/python/ccxt/allcoin.py b/python/ccxt/allcoin.py index 337fb2567aabb..0df46ce169798 100644 --- a/python/ccxt/allcoin.py +++ b/python/ccxt/allcoin.py @@ -10,7 +10,9 @@ def describe(self): 'id': 'allcoin', 'name': 'Allcoin', 'countries': 'CA', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'extension': '', 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/31561809-c316b37c-b061-11e7-8d5a-b547b4d730eb.jpg', diff --git a/python/ccxt/anxpro.py b/python/ccxt/anxpro.py index 41bd12172eebd..09232e3e8234f 100644 --- a/python/ccxt/anxpro.py +++ b/python/ccxt/anxpro.py @@ -15,9 +15,11 @@ def describe(self): 'countries': ['JP', 'SG', 'HK', 'NZ'], 'version': '2', 'rateLimit': 1500, - 'hasCORS': False, - 'hasFetchTrades': False, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'fetchTrades': False, + 'withdraw': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27765983-fd8595da-5ec9-11e7-82e3-adb3ab8c2612.jpg', 'api': 'https://anxpro.com/api', diff --git a/python/ccxt/async/_1broker.py b/python/ccxt/async/_1broker.py index 80d9314d590e4..04d61967be566 100644 --- a/python/ccxt/async/_1broker.py +++ b/python/ccxt/async/_1broker.py @@ -13,10 +13,12 @@ def describe(self): 'countries': 'US', 'rateLimit': 1500, 'version': 'v2', - 'hasPublicAPI': False, - 'hasCORS': True, - 'hasFetchTrades': False, - 'hasFetchOHLCV': True, + 'has': { + 'publicAPI': False, + 'CORS': True, + 'fetchTrades': False, + 'fetchOHLCV': True, + }, 'timeframes': { '1m': '60', '15m': '900', diff --git a/python/ccxt/async/_1btcxe.py b/python/ccxt/async/_1btcxe.py index 7485fd15a17c1..ccbe1bcd1e3eb 100644 --- a/python/ccxt/async/_1btcxe.py +++ b/python/ccxt/async/_1btcxe.py @@ -12,9 +12,11 @@ def describe(self): 'name': '1BTCXE', 'countries': 'PA', # Panama 'comment': 'Crypto Capital API', - 'hasCORS': True, - 'hasFetchOHLCV': True, - 'hasWithdraw': True, + 'has': { + 'CORS': True, + 'fetchTickers': True, + 'withdraw': True, + }, 'timeframes': { '1d': '1year', }, diff --git a/python/ccxt/async/acx.py b/python/ccxt/async/acx.py index 09f2e01fa3fe9..3b88c165277d5 100644 --- a/python/ccxt/async/acx.py +++ b/python/ccxt/async/acx.py @@ -14,10 +14,12 @@ def describe(self): 'countries': 'AU', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': True, - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, - 'hasWithdraw': True, + 'has': { + 'CORS': True, + 'fetchTickers': True, + 'fetchOHLCV': True, + 'withdraw': True, + }, 'timeframes': { '1m': '1', '5m': '5', diff --git a/python/ccxt/async/allcoin.py b/python/ccxt/async/allcoin.py index d1957b730a41b..322750b9fac7c 100644 --- a/python/ccxt/async/allcoin.py +++ b/python/ccxt/async/allcoin.py @@ -10,7 +10,9 @@ def describe(self): 'id': 'allcoin', 'name': 'Allcoin', 'countries': 'CA', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'extension': '', 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/31561809-c316b37c-b061-11e7-8d5a-b547b4d730eb.jpg', diff --git a/python/ccxt/async/anxpro.py b/python/ccxt/async/anxpro.py index c1fd3ab0f998c..520b89d004cb4 100644 --- a/python/ccxt/async/anxpro.py +++ b/python/ccxt/async/anxpro.py @@ -15,9 +15,11 @@ def describe(self): 'countries': ['JP', 'SG', 'HK', 'NZ'], 'version': '2', 'rateLimit': 1500, - 'hasCORS': False, - 'hasFetchTrades': False, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'fetchTrades': False, + 'withdraw': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27765983-fd8595da-5ec9-11e7-82e3-adb3ab8c2612.jpg', 'api': 'https://anxpro.com/api', diff --git a/python/ccxt/async/bibox.py b/python/ccxt/async/bibox.py index c458127b3078c..c3b2f5625fa33 100644 --- a/python/ccxt/async/bibox.py +++ b/python/ccxt/async/bibox.py @@ -16,16 +16,9 @@ def describe(self): 'name': 'Bibox', 'countries': ['CN', 'US', 'KR'], 'version': 'v1', - 'hasCORS': False, - 'hasPublicAPI': False, - 'hasFetchBalance': True, - 'hasFetchCurrencies': True, - 'hasFetchTickers': True, - 'hasFetchOrders': True, - 'hasFetchMyTrades': True, - 'hasFetchOHLCV': True, - 'hasWithdraw': True, 'has': { + 'CORS': False, + 'publicAPI': False, 'fetchBalance': True, 'fetchCurrencies': True, 'fetchTickers': True, diff --git a/python/ccxt/async/binance.py b/python/ccxt/async/binance.py index 7d89b2d4ae5ed..e3d018e1cab1f 100644 --- a/python/ccxt/async/binance.py +++ b/python/ccxt/async/binance.py @@ -18,18 +18,9 @@ def describe(self): 'name': 'Binance', 'countries': 'JP', # Japan 'rateLimit': 500, - 'hasCORS': False, - # obsolete metainfo interface - 'hasFetchBidsAsks': True, - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, - 'hasFetchMyTrades': True, - 'hasFetchOrder': True, - 'hasFetchOrders': True, - 'hasFetchOpenOrders': True, - 'hasWithdraw': True, # new metainfo interface 'has': { + 'CORS': False, 'fetchBidsAsks': True, 'fetchTickers': True, 'fetchOHLCV': True, diff --git a/python/ccxt/async/bit2c.py b/python/ccxt/async/bit2c.py index db60e3c0891d7..657330961c623 100644 --- a/python/ccxt/async/bit2c.py +++ b/python/ccxt/async/bit2c.py @@ -12,7 +12,9 @@ def describe(self): 'name': 'Bit2C', 'countries': 'IL', # Israel 'rateLimit': 3000, - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766119-3593220e-5ece-11e7-8b3a-5a041f6bcc3f.jpg', 'api': 'https://www.bit2c.co.il', diff --git a/python/ccxt/async/bitbay.py b/python/ccxt/async/bitbay.py index 9c6a70476d0ac..1b48eeffce6a7 100644 --- a/python/ccxt/async/bitbay.py +++ b/python/ccxt/async/bitbay.py @@ -27,8 +27,10 @@ def describe(self): 'name': 'BitBay', 'countries': ['PL', 'EU'], # Poland 'rateLimit': 1000, - 'hasCORS': True, - 'hasWithdraw': True, + 'has': { + 'CORS': True, + 'withdraw': True + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766132-978a7bd8-5ece-11e7-9540-bc96d1e9bbb8.jpg', 'www': 'https://bitbay.net', diff --git a/python/ccxt/async/bitcoincoid.py b/python/ccxt/async/bitcoincoid.py index 06e7434abd632..3d9b6c455b253 100644 --- a/python/ccxt/async/bitcoincoid.py +++ b/python/ccxt/async/bitcoincoid.py @@ -12,19 +12,8 @@ def describe(self): 'id': 'bitcoincoid', 'name': 'Bitcoin.co.id', 'countries': 'ID', # Indonesia - 'hasCORS': False, - # obsolete metainfo interface - 'hasFetchTickers': False, - 'hasFetchOHLCV': False, - 'hasFetchOrder': True, - 'hasFetchOrders': False, - 'hasFetchClosedOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchMyTrades': False, - 'hasFetchCurrencies': False, - 'hasWithdraw': False, - # new metainfo interface 'has': { + 'CORS': False, 'fetchTickers': False, 'fetchOHLCV': False, 'fetchOrder': True, diff --git a/python/ccxt/async/bitfinex.py b/python/ccxt/async/bitfinex.py index d0bba87f01ada..ee1971a095b7e 100644 --- a/python/ccxt/async/bitfinex.py +++ b/python/ccxt/async/bitfinex.py @@ -23,17 +23,9 @@ def describe(self): 'countries': 'VG', 'version': 'v1', 'rateLimit': 1500, - 'hasCORS': False, - # old metainfo interface - 'hasFetchOrder': True, - 'hasFetchTickers': True, - 'hasDeposit': True, - 'hasWithdraw': True, - 'hasFetchOHLCV': True, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, # new metainfo interface 'has': { + 'CORS': False, 'fetchOHLCV': True, 'fetchTickers': True, 'fetchOrder': True, diff --git a/python/ccxt/async/bitfinex2.py b/python/ccxt/async/bitfinex2.py index 3873d1191354b..3aaeac726d34a 100644 --- a/python/ccxt/async/bitfinex2.py +++ b/python/ccxt/async/bitfinex2.py @@ -15,18 +15,9 @@ def describe(self): 'name': 'Bitfinex v2', 'countries': 'VG', 'version': 'v2', - 'hasCORS': True, - # old metainfo interface - 'hasCreateOrder': False, - 'hasFetchOrder': True, - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, - 'hasWithdraw': True, - 'hasDeposit': False, - 'hasFetchOpenOrders': False, - 'hasFetchClosedOrders': False, # new metainfo interface 'has': { + 'CORS': True, 'createOrder': False, 'fetchOHLCV': True, 'fetchTickers': True, diff --git a/python/ccxt/async/bitflyer.py b/python/ccxt/async/bitflyer.py index 2e497965d5612..c51821b3c83f0 100644 --- a/python/ccxt/async/bitflyer.py +++ b/python/ccxt/async/bitflyer.py @@ -12,8 +12,10 @@ def describe(self): 'countries': 'JP', 'version': 'v1', 'rateLimit': 500, - 'hasCORS': False, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'withdraw': True + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28051642-56154182-660e-11e7-9b0d-6042d1e6edd8.jpg', 'api': 'https://api.bitflyer.jp', diff --git a/python/ccxt/async/bithumb.py b/python/ccxt/async/bithumb.py index b8c917935fb38..b6145b47b8083 100644 --- a/python/ccxt/async/bithumb.py +++ b/python/ccxt/async/bithumb.py @@ -14,12 +14,8 @@ def describe(self): 'name': 'Bithumb', 'countries': 'KR', # South Korea 'rateLimit': 500, - 'hasCORS': True, - # obsolete metainfo interface - 'hasFetchTickers': True, - 'hasWithdraw': True, - # new metainfo interface 'has': { + 'CORS': True, 'fetchTickers': True, 'withdraw': True, }, diff --git a/python/ccxt/async/bitlish.py b/python/ccxt/async/bitlish.py index 6aa53ac4f86c5..ceeb6e8f13873 100644 --- a/python/ccxt/async/bitlish.py +++ b/python/ccxt/async/bitlish.py @@ -13,10 +13,12 @@ def describe(self): 'countries': ['GB', 'EU', 'RU'], 'rateLimit': 1500, 'version': 'v1', - 'hasCORS': False, - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'fetchTickers': True, + 'fetchOHLCV': True, + 'withdraw': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766275-dcfc6c30-5ed3-11e7-839d-00a846385d0b.jpg', 'api': 'https://bitlish.com/api', diff --git a/python/ccxt/async/bitmarket.py b/python/ccxt/async/bitmarket.py index b2a7f9f6d12ad..6e7f5f4cd1666 100644 --- a/python/ccxt/async/bitmarket.py +++ b/python/ccxt/async/bitmarket.py @@ -13,9 +13,11 @@ def describe(self): 'name': 'BitMarket', 'countries': ['PL', 'EU'], 'rateLimit': 1500, - 'hasCORS': False, - 'hasFetchOHLCV': True, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'fetchOHLCV': True, + 'withdraw': True, + }, 'timeframes': { '90m': '90m', '6h': '6h', diff --git a/python/ccxt/async/bitmex.py b/python/ccxt/async/bitmex.py index 40917204e7c4e..b5fe78d65aff7 100644 --- a/python/ccxt/async/bitmex.py +++ b/python/ccxt/async/bitmex.py @@ -16,9 +16,11 @@ def describe(self): 'version': 'v1', 'userAgent': None, 'rateLimit': 1500, - 'hasCORS': False, - 'hasFetchOHLCV': True, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'fetchOHLCV': True, + 'withdraw': True, + }, 'timeframes': { '1m': '1m', '5m': '5m', diff --git a/python/ccxt/async/bitso.py b/python/ccxt/async/bitso.py index 79815d945c947..a010372b07579 100644 --- a/python/ccxt/async/bitso.py +++ b/python/ccxt/async/bitso.py @@ -13,7 +13,9 @@ def describe(self): 'countries': 'MX', # Mexico 'rateLimit': 2000, # 30 requests per minute 'version': 'v3', - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766335-715ce7aa-5ed5-11e7-88a8-173a27bb30fe.jpg', 'api': 'https://api.bitso.com', diff --git a/python/ccxt/async/bitstamp.py b/python/ccxt/async/bitstamp.py index 3da180f8c6db3..47f4733cac044 100644 --- a/python/ccxt/async/bitstamp.py +++ b/python/ccxt/async/bitstamp.py @@ -14,12 +14,8 @@ def describe(self): 'countries': 'GB', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': False, - # obsolete metainfo interface - 'hasFetchOrder': True, - 'hasWithdraw': True, - # new metainfo interface 'has': { + 'CORS': True, 'fetchOrder': True, 'withdraw': True, }, diff --git a/python/ccxt/async/bitstamp1.py b/python/ccxt/async/bitstamp1.py index f7156beba4fa4..a287b5c42389e 100644 --- a/python/ccxt/async/bitstamp1.py +++ b/python/ccxt/async/bitstamp1.py @@ -14,7 +14,9 @@ def describe(self): 'countries': 'GB', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27786377-8c8ab57e-5fe9-11e7-8ea4-2b05b6bcceec.jpg', 'api': 'https://www.bitstamp.net/api', diff --git a/python/ccxt/async/bittrex.py b/python/ccxt/async/bittrex.py index 0f8e62898d949..5f62f14135ba0 100644 --- a/python/ccxt/async/bittrex.py +++ b/python/ccxt/async/bittrex.py @@ -31,19 +31,9 @@ def describe(self): 'version': 'v1.1', 'rateLimit': 1500, 'hasAlreadyAuthenticatedSuccessfully': False, # a workaround for APIKEY_INVALID - 'hasCORS': False, - # obsolete metainfo interface - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, - 'hasFetchOrder': True, - 'hasFetchOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchMyTrades': False, - 'hasFetchCurrencies': True, - 'hasWithdraw': True, # new metainfo interface 'has': { + 'CORS': True, 'fetchTickers': True, 'fetchOHLCV': True, 'fetchOrder': True, diff --git a/python/ccxt/async/bl3p.py b/python/ccxt/async/bl3p.py index e2652d7227d96..760d3d58f4cf8 100644 --- a/python/ccxt/async/bl3p.py +++ b/python/ccxt/async/bl3p.py @@ -15,7 +15,9 @@ def describe(self): 'rateLimit': 1000, 'version': '1', 'comment': 'An exchange market by BitonicNL', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28501752-60c21b82-6feb-11e7-818b-055ee6d0e754.jpg', 'api': 'https://api.bl3p.eu', diff --git a/python/ccxt/async/bleutrade.py b/python/ccxt/async/bleutrade.py index a85aa13ac472c..5d5775d923033 100644 --- a/python/ccxt/async/bleutrade.py +++ b/python/ccxt/async/bleutrade.py @@ -17,9 +17,11 @@ def describe(self): 'countries': 'BR', # Brazil 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': True, - 'hasFetchTickers': True, - 'hasFetchOHLCV': False, + 'has': { + 'CORS': True, + 'fetchTickers': True, + 'fetchOHLCV': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30303000-b602dbe6-976d-11e7-956d-36c5049c01e7.jpg', 'api': { diff --git a/python/ccxt/async/braziliex.py b/python/ccxt/async/braziliex.py index 4048a63e425fb..a5d5578fbcac3 100644 --- a/python/ccxt/async/braziliex.py +++ b/python/ccxt/async/braziliex.py @@ -16,11 +16,6 @@ def describe(self): 'name': 'Braziliex', 'countries': 'BR', 'rateLimit': 1000, - # obsolete metainfo interface - 'hasFetchTickers': True, - 'hasFetchOpenOrders': True, - 'hasFetchMyTrades': True, - # new metainfo interface 'has': { 'fetchTickers': True, 'fetchOpenOrders': True, diff --git a/python/ccxt/async/btcbox.py b/python/ccxt/async/btcbox.py index 6221c2946546f..47eff731fe909 100644 --- a/python/ccxt/async/btcbox.py +++ b/python/ccxt/async/btcbox.py @@ -13,8 +13,10 @@ def describe(self): 'countries': 'JP', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': False, - 'hasFetchOHLCV': False, + 'has': { + 'CORS': False, + 'fetchOHLCV': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/31275803-4df755a8-aaa1-11e7-9abb-11ec2fad9f2d.jpg', 'api': 'https://www.btcbox.co.jp/api', diff --git a/python/ccxt/async/btcchina.py b/python/ccxt/async/btcchina.py index 172cb50673d52..c8cb05bfd085d 100644 --- a/python/ccxt/async/btcchina.py +++ b/python/ccxt/async/btcchina.py @@ -14,7 +14,9 @@ def describe(self): 'countries': 'CN', 'rateLimit': 1500, 'version': 'v1', - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766368-465b3286-5ed6-11e7-9a11-0f6467e1d82b.jpg', 'api': { diff --git a/python/ccxt/async/btcexchange.py b/python/ccxt/async/btcexchange.py index 6b635aa799426..28cfd7fa9dc48 100644 --- a/python/ccxt/async/btcexchange.py +++ b/python/ccxt/async/btcexchange.py @@ -11,7 +11,9 @@ def describe(self): 'name': 'BTCExchange', 'countries': 'PH', # Philippines 'rateLimit': 1500, - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27993052-4c92911a-64aa-11e7-96d8-ec6ac3435757.jpg', 'api': 'https://www.btcexchange.ph/api', diff --git a/python/ccxt/async/btcmarkets.py b/python/ccxt/async/btcmarkets.py index c4b1c390769b8..79937bbd9a96a 100644 --- a/python/ccxt/async/btcmarkets.py +++ b/python/ccxt/async/btcmarkets.py @@ -16,13 +16,8 @@ def describe(self): 'name': 'BTC Markets', 'countries': 'AU', # Australia 'rateLimit': 1000, # market data cached for 1 second(trades cached for 2 seconds) - 'hasCORS': False, - 'hasFetchOrder': True, - 'hasFetchOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchMyTrades': True, 'has': { + 'CORS': False, 'fetchOrder': True, 'fetchOrders': True, 'fetchClosedOrders': 'emulated', diff --git a/python/ccxt/async/btctradeua.py b/python/ccxt/async/btctradeua.py index 3f72f352d7082..848a2465dd689 100644 --- a/python/ccxt/async/btctradeua.py +++ b/python/ccxt/async/btctradeua.py @@ -12,7 +12,9 @@ def describe(self): 'name': 'BTC Trade UA', 'countries': 'UA', # Ukraine, 'rateLimit': 3000, - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27941483-79fc7350-62d9-11e7-9f61-ac47f28fcd96.jpg', 'api': 'https://btc-trade.com.ua/api', diff --git a/python/ccxt/async/btcturk.py b/python/ccxt/async/btcturk.py index b1f60b1b80fd8..8839a4f106c05 100644 --- a/python/ccxt/async/btcturk.py +++ b/python/ccxt/async/btcturk.py @@ -14,9 +14,11 @@ def describe(self): 'name': 'BTCTurk', 'countries': 'TR', # Turkey 'rateLimit': 1000, - 'hasCORS': True, - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, + 'has': { + 'CORS': True, + 'fetchTickers': True, + 'fetchOHLCV': True, + }, 'timeframes': { '1d': '1d', }, diff --git a/python/ccxt/async/btcx.py b/python/ccxt/async/btcx.py index 00a44241e59e5..6bd75614ecbd2 100644 --- a/python/ccxt/async/btcx.py +++ b/python/ccxt/async/btcx.py @@ -14,7 +14,9 @@ def describe(self): 'countries': ['IS', 'US', 'EU'], 'rateLimit': 1500, # support in english is very poor, unable to tell rate limits 'version': 'v1', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766385-9fdcc98c-5ed6-11e7-8f14-66d5e5cd47e6.jpg', 'api': 'https://btc-x.is/api', diff --git a/python/ccxt/async/bter.py b/python/ccxt/async/bter.py index 3f88b1cdd078f..81899dfc76949 100644 --- a/python/ccxt/async/bter.py +++ b/python/ccxt/async/bter.py @@ -13,11 +13,6 @@ def describe(self): 'name': 'Bter', 'countries': ['VG', 'CN'], # British Virgin Islands, China 'version': '2', - # obsolete metainfo interface - 'hasCORS': False, - 'hasFetchTickers': True, - 'hasWithdraw': True, - # new metainfo interface 'has': { 'CORS': False, 'fetchTickers': True, diff --git a/python/ccxt/async/bxinth.py b/python/ccxt/async/bxinth.py index 40c45cb07db00..c052cfe8258a7 100644 --- a/python/ccxt/async/bxinth.py +++ b/python/ccxt/async/bxinth.py @@ -12,8 +12,10 @@ def describe(self): 'name': 'BX.in.th', 'countries': 'TH', # Thailand 'rateLimit': 1500, - 'hasCORS': False, - 'hasFetchTickers': True, + 'has': { + 'CORS': False, + 'fetchTickers': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766412-567b1eb4-5ed7-11e7-94a8-ff6a3884f6c5.jpg', 'api': 'https://bx.in.th/api', diff --git a/python/ccxt/async/ccex.py b/python/ccxt/async/ccex.py index 7e9e69036a7e8..214612fac5564 100644 --- a/python/ccxt/async/ccex.py +++ b/python/ccxt/async/ccex.py @@ -13,8 +13,10 @@ def describe(self): 'name': 'C-CEX', 'countries': ['DE', 'EU'], 'rateLimit': 1500, - 'hasCORS': False, - 'hasFetchTickers': True, + 'has': { + 'CORS': False, + 'fetchTickers': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766433-16881f90-5ed8-11e7-92f8-3d92cc747a6c.jpg', 'api': { diff --git a/python/ccxt/async/cex.py b/python/ccxt/async/cex.py index 7105cc3cc5eeb..f6e671c81f228 100644 --- a/python/ccxt/async/cex.py +++ b/python/ccxt/async/cex.py @@ -15,10 +15,12 @@ def describe(self): 'name': 'CEX.IO', 'countries': ['GB', 'EU', 'CY', 'RU'], 'rateLimit': 1500, - 'hasCORS': True, - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, - 'hasFetchOpenOrders': True, + 'has': { + 'CORS': True, + 'fetchTickers': True, + 'fetchOHLCV': True, + 'fetchOpenOrders': True, + }, 'timeframes': { '1m': '1m', }, diff --git a/python/ccxt/async/chbtc.py b/python/ccxt/async/chbtc.py index 98aa1133eb95f..60c2c70a5e63d 100644 --- a/python/ccxt/async/chbtc.py +++ b/python/ccxt/async/chbtc.py @@ -13,8 +13,10 @@ def describe(self): 'countries': 'CN', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': False, - 'hasFetchOrder': True, + 'has': { + 'CORS': False, + 'fetchOrder': True + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28555659-f0040dc2-7109-11e7-9d99-688a438bf9f4.jpg', 'api': { diff --git a/python/ccxt/async/chilebit.py b/python/ccxt/async/chilebit.py index fd28f94a7fb87..9405a560ffb81 100644 --- a/python/ccxt/async/chilebit.py +++ b/python/ccxt/async/chilebit.py @@ -10,7 +10,9 @@ def describe(self): 'id': 'chilebit', 'name': 'ChileBit', 'countries': 'CL', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27991414-1298f0d8-647f-11e7-9c40-d56409266336.jpg', 'api': { diff --git a/python/ccxt/async/coincheck.py b/python/ccxt/async/coincheck.py index eed39382937b6..23f696ad8a1a2 100644 --- a/python/ccxt/async/coincheck.py +++ b/python/ccxt/async/coincheck.py @@ -13,7 +13,9 @@ def describe(self): 'name': 'coincheck', 'countries': ['JP', 'ID'], 'rateLimit': 1500, - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766464-3b5c3c74-5ed9-11e7-840e-31b32968e1da.jpg', 'api': 'https://coincheck.com/api', diff --git a/python/ccxt/async/coinexchange.py b/python/ccxt/async/coinexchange.py index 98c3fe1187209..f723f1da4d9a4 100644 --- a/python/ccxt/async/coinexchange.py +++ b/python/ccxt/async/coinexchange.py @@ -13,13 +13,9 @@ def describe(self): 'name': 'CoinExchange', 'countries': ['IN', 'JP', 'KR', 'VN', 'US'], 'rateLimit': 1000, - # obsolete metainfo interface - 'hasPrivateAPI': False, - 'hasFetchTrades': False, - 'hasFetchCurrencies': True, - 'hasFetchTickers': True, # new metainfo interface 'has': { + 'privateAPI': False, 'fetchTrades': False, 'fetchCurrencies': True, 'fetchTickers': True, diff --git a/python/ccxt/async/coinfloor.py b/python/ccxt/async/coinfloor.py index 0463142f75880..5e2ae3fd8da55 100644 --- a/python/ccxt/async/coinfloor.py +++ b/python/ccxt/async/coinfloor.py @@ -13,7 +13,9 @@ def describe(self): 'name': 'coinfloor', 'rateLimit': 1000, 'countries': 'UK', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28246081-623fc164-6a1c-11e7-913f-bac0d5576c90.jpg', 'api': 'https://webapi.coinfloor.co.uk:8090/bist', diff --git a/python/ccxt/async/coingi.py b/python/ccxt/async/coingi.py index fd755daec41a0..ac04ba3a805fe 100644 --- a/python/ccxt/async/coingi.py +++ b/python/ccxt/async/coingi.py @@ -22,8 +22,10 @@ def describe(self): 'name': 'Coingi', 'rateLimit': 1000, 'countries': ['PA', 'BG', 'CN', 'US'], # Panama, Bulgaria, China, US - 'hasFetchTickers': True, - 'hasCORS': False, + 'has': { + 'CORS': False, + 'fetchTickers': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28619707-5c9232a8-7212-11e7-86d6-98fe5d15cc6e.jpg', 'api': { diff --git a/python/ccxt/async/coinmarketcap.py b/python/ccxt/async/coinmarketcap.py index cdf6256ae26db..c3ec3292ac41e 100644 --- a/python/ccxt/async/coinmarketcap.py +++ b/python/ccxt/async/coinmarketcap.py @@ -14,16 +14,16 @@ def describe(self): 'rateLimit': 10000, 'version': 'v1', 'countries': 'US', - 'hasCORS': True, - 'hasPrivateAPI': False, - 'hasCreateOrder': False, - 'hasCancelOrder': False, - 'hasFetchBalance': False, - 'hasFetchOrderBook': False, - 'hasFetchTrades': False, - 'hasFetchTickers': True, - 'hasFetchCurrencies': True, 'has': { + 'CORS': True, + 'privateAPI': False, + 'createOrder': False, + 'cancelOrder': False, + 'fetchBalance': False, + 'fetchOrderBook': False, + 'fetchTrades': False, + 'fetchTickers': True, + 'fetchCurrencies': True, 'fetchCurrencies': True, }, 'urls': { diff --git a/python/ccxt/async/coinmate.py b/python/ccxt/async/coinmate.py index 31b0b4aca9f2f..fb80c00ee57d3 100644 --- a/python/ccxt/async/coinmate.py +++ b/python/ccxt/async/coinmate.py @@ -12,7 +12,9 @@ def describe(self): 'name': 'CoinMate', 'countries': ['GB', 'CZ', 'EU'], # UK, Czech Republic 'rateLimit': 1000, - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27811229-c1efb510-606c-11e7-9a36-84ba2ce412d8.jpg', 'api': 'https://coinmate.io/api', diff --git a/python/ccxt/async/coinsecure.py b/python/ccxt/async/coinsecure.py index 16467ad4dbbc7..aecfc15f9e33e 100644 --- a/python/ccxt/async/coinsecure.py +++ b/python/ccxt/async/coinsecure.py @@ -14,7 +14,9 @@ def describe(self): 'countries': 'IN', # India 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766472-9cbd200a-5ed9-11e7-9551-2267ad7bac08.jpg', 'api': 'https://api.coinsecure.in', diff --git a/python/ccxt/async/coinspot.py b/python/ccxt/async/coinspot.py index 340f5d2a578a2..5e66d48e537e9 100644 --- a/python/ccxt/async/coinspot.py +++ b/python/ccxt/async/coinspot.py @@ -14,7 +14,9 @@ def describe(self): 'name': 'CoinSpot', 'countries': 'AU', # Australia 'rateLimit': 1000, - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28208429-3cacdf9a-6896-11e7-854e-4c79a772a30f.jpg', 'api': { diff --git a/python/ccxt/async/cryptopia.py b/python/ccxt/async/cryptopia.py index dc1bc0c40baa6..47284b2312c45 100644 --- a/python/ccxt/async/cryptopia.py +++ b/python/ccxt/async/cryptopia.py @@ -18,19 +18,8 @@ def describe(self): 'name': 'Cryptopia', 'rateLimit': 1500, 'countries': 'NZ', # New Zealand - 'hasCORS': False, - # obsolete metainfo interface - 'hasFetchTickers': True, - 'hasFetchOrder': True, - 'hasFetchOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchMyTrades': True, - 'hasFetchCurrencies': True, - 'hasDeposit': True, - 'hasWithdraw': True, - # new metainfo interface 'has': { + 'CORS': False, 'fetchTickers': True, 'fetchOrder': 'emulated', 'fetchOrders': 'emulated', diff --git a/python/ccxt/async/dsx.py b/python/ccxt/async/dsx.py index 37da4e4ce9eeb..8131ee2955d50 100644 --- a/python/ccxt/async/dsx.py +++ b/python/ccxt/async/dsx.py @@ -12,13 +12,15 @@ def describe(self): 'name': 'DSX', 'countries': 'UK', 'rateLimit': 1500, - 'hasCORS': False, - 'hasFetchOrder': True, - 'hasFetchOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchTickers': True, - 'hasFetchMyTrades': True, + 'has': { + 'CORS': False, + 'fetchOrder': True, + 'fetchOrders': True, + 'fetchOpenOrders': True, + 'fetchClosedOrders': True, + 'fetchTickers': True, + 'fetchMyTrades': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27990275-1413158a-645a-11e7-931c-94717f7510e3.jpg', 'api': { diff --git a/python/ccxt/async/exmo.py b/python/ccxt/async/exmo.py index a16398526fcc3..e934c06a6c08d 100644 --- a/python/ccxt/async/exmo.py +++ b/python/ccxt/async/exmo.py @@ -14,9 +14,11 @@ def describe(self): 'countries': ['ES', 'RU'], # Spain, Russia 'rateLimit': 1000, # once every 350 ms ≈ 180 requests per minute ≈ 3 requests per second 'version': 'v1', - 'hasCORS': False, - 'hasFetchTickers': True, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'fetchTickers': True, + 'withdraw': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766491-1b0ea956-5eda-11e7-9225-40d67b481b8d.jpg', 'api': 'https://api.exmo.com', diff --git a/python/ccxt/async/flowbtc.py b/python/ccxt/async/flowbtc.py index bf07f5f5a39d9..d5247984c5197 100644 --- a/python/ccxt/async/flowbtc.py +++ b/python/ccxt/async/flowbtc.py @@ -13,7 +13,9 @@ def describe(self): 'countries': 'BR', # Brazil 'version': 'v1', 'rateLimit': 1000, - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28162465-cd815d4c-67cf-11e7-8e57-438bea0523a2.jpg', 'api': 'https://api.flowbtc.com:8400/ajax', diff --git a/python/ccxt/async/foxbit.py b/python/ccxt/async/foxbit.py index 778d043a7380c..763efc7cdd795 100644 --- a/python/ccxt/async/foxbit.py +++ b/python/ccxt/async/foxbit.py @@ -11,7 +11,9 @@ def describe(self): 'id': 'foxbit', 'name': 'FoxBit', 'countries': 'BR', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'rateLimit': 1000, 'version': 'v1', 'urls': { diff --git a/python/ccxt/async/fybse.py b/python/ccxt/async/fybse.py index e6a9f8e2b68cf..4bb2abb390e9d 100644 --- a/python/ccxt/async/fybse.py +++ b/python/ccxt/async/fybse.py @@ -12,7 +12,9 @@ def describe(self): 'id': 'fybse', 'name': 'FYB-SE', 'countries': 'SE', # Sweden - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'rateLimit': 1500, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766512-31019772-5edb-11e7-8241-2e675e6797f1.jpg', diff --git a/python/ccxt/async/fybsg.py b/python/ccxt/async/fybsg.py index e0c1389b797c0..bc0db0ce23a3c 100644 --- a/python/ccxt/async/fybsg.py +++ b/python/ccxt/async/fybsg.py @@ -10,7 +10,9 @@ def describe(self): 'id': 'fybsg', 'name': 'FYB-SG', 'countries': 'SG', # Singapore - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766513-3364d56a-5edb-11e7-9e6b-d5898bb89c81.jpg', 'api': 'https://www.fybsg.com/api/SGD', diff --git a/python/ccxt/async/gatecoin.py b/python/ccxt/async/gatecoin.py index 799f683b49334..8d0d9f2c8b49b 100644 --- a/python/ccxt/async/gatecoin.py +++ b/python/ccxt/async/gatecoin.py @@ -15,9 +15,11 @@ def describe(self): 'rateLimit': 2000, 'countries': 'HK', # Hong Kong 'comment': 'a regulated/licensed exchange', - 'hasCORS': False, - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, + 'has': { + 'CORS': False, + 'fetchTickers': True, + 'fetchOHLCV': True, + }, 'timeframes': { '1m': '1m', '15m': '15m', diff --git a/python/ccxt/async/gdax.py b/python/ccxt/async/gdax.py index 02ca4ea32b122..77cd0ead847cf 100644 --- a/python/ccxt/async/gdax.py +++ b/python/ccxt/async/gdax.py @@ -20,16 +20,6 @@ def describe(self): 'countries': 'US', 'rateLimit': 1000, 'userAgent': self.userAgents['chrome'], - # obsolete metainfo interface - 'hasCORS': True, - 'hasFetchOHLCV': True, - 'hasDeposit': True, - 'hasWithdraw': True, - 'hasFetchOrder': True, - 'hasFetchOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - # new metainfo interface 'has': { 'CORS': True, 'fetchOHLCV': True, diff --git a/python/ccxt/async/gemini.py b/python/ccxt/async/gemini.py index 131e712a567e6..f417146d88852 100644 --- a/python/ccxt/async/gemini.py +++ b/python/ccxt/async/gemini.py @@ -15,10 +15,6 @@ def describe(self): 'countries': 'US', 'rateLimit': 1500, # 200 for private API 'version': 'v1', - # obsolete metainfo interface - 'hasCORS': False, - 'hasWithdraw': True, - # new metainfo interface 'has': { 'CORS': False, 'withdraw': True, diff --git a/python/ccxt/async/hitbtc.py b/python/ccxt/async/hitbtc.py index 36459249f8cab..f00cd1dc22ea8 100644 --- a/python/ccxt/async/hitbtc.py +++ b/python/ccxt/async/hitbtc.py @@ -16,12 +16,13 @@ def describe(self): 'countries': 'UK', 'rateLimit': 1500, 'version': '1', - 'hasCORS': False, - 'hasFetchTickers': True, - 'hasFetchOrder': True, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'fetchOrder': True, + 'fetchOpenOrders': True, + 'fetchClosedOrders': True, + 'withdraw': True + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766555-8eaec20e-5edc-11e7-9c5b-6dc69fc42f5e.jpg', 'api': 'http://api.hitbtc.com', diff --git a/python/ccxt/async/hitbtc2.py b/python/ccxt/async/hitbtc2.py index 252bf0d2e6816..aa22f91cc257f 100644 --- a/python/ccxt/async/hitbtc2.py +++ b/python/ccxt/async/hitbtc2.py @@ -19,19 +19,8 @@ def describe(self): 'countries': 'UK', 'rateLimit': 1500, 'version': '2', - 'hasCORS': True, - # older metainfo interface - 'hasFetchOHLCV': True, - 'hasFetchTickers': True, - 'hasFetchOrder': True, - 'hasFetchOrders': False, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchMyTrades': True, - 'hasWithdraw': True, - 'hasFetchCurrencies': True, - # new metainfo interface 'has': { + 'CORS': True, 'fetchCurrencies': True, 'fetchOHLCV': True, 'fetchTickers': True, diff --git a/python/ccxt/async/huobi.py b/python/ccxt/async/huobi.py index dd365cc89baa7..d3112aa727aef 100644 --- a/python/ccxt/async/huobi.py +++ b/python/ccxt/async/huobi.py @@ -13,8 +13,10 @@ def describe(self): 'countries': 'CN', 'rateLimit': 2000, 'version': 'v3', - 'hasCORS': False, - 'hasFetchOHLCV': True, + 'has': { + 'CORS': False, + 'fetchOHLCV': True, + }, 'timeframes': { '1m': '001', '5m': '005', diff --git a/python/ccxt/async/huobicny.py b/python/ccxt/async/huobicny.py index 3d6b08d867940..19fcb7af09ed6 100644 --- a/python/ccxt/async/huobicny.py +++ b/python/ccxt/async/huobicny.py @@ -10,7 +10,9 @@ def describe(self): 'id': 'huobicny', 'name': 'Huobi CNY', 'hostname': 'be.huobi.com', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766569-15aa7b9a-5edd-11e7-9e7f-44791f4ee49c.jpg', 'api': 'https://be.huobi.com', diff --git a/python/ccxt/async/huobipro.py b/python/ccxt/async/huobipro.py index 887a7112e8840..60c5b0960ef9e 100644 --- a/python/ccxt/async/huobipro.py +++ b/python/ccxt/async/huobipro.py @@ -19,13 +19,8 @@ def describe(self): 'accounts': None, 'accountsById': None, 'hostname': 'api.huobi.pro', - 'hasCORS': False, - # obsolete metainfo structure - 'hasFetchOHLCV': True, - 'hasFetchOrders': True, - 'hasFetchOpenOrders': True, - # new metainfo structure 'has': { + 'CORS': False, 'fetchOHCLV': True, 'fetchOrders': True, 'fetchOpenOrders': True, diff --git a/python/ccxt/async/independentreserve.py b/python/ccxt/async/independentreserve.py index 6a3cad3722d06..950614a76888b 100644 --- a/python/ccxt/async/independentreserve.py +++ b/python/ccxt/async/independentreserve.py @@ -11,7 +11,9 @@ def describe(self): 'name': 'Independent Reserve', 'countries': ['AU', 'NZ'], # Australia, New Zealand 'rateLimit': 1000, - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30521662-cf3f477c-9bcb-11e7-89bc-d1ac85012eda.jpg', 'api': { diff --git a/python/ccxt/async/itbit.py b/python/ccxt/async/itbit.py index 696ca2af7e02d..724b6625dad9e 100644 --- a/python/ccxt/async/itbit.py +++ b/python/ccxt/async/itbit.py @@ -14,7 +14,9 @@ def describe(self): 'countries': 'US', 'rateLimit': 2000, 'version': 'v1', - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27822159-66153620-60ad-11e7-89e7-005f6d7f3de0.jpg', 'api': 'https://api.itbit.com', diff --git a/python/ccxt/async/jubi.py b/python/ccxt/async/jubi.py index 658f0022a7fe4..004e07132c421 100644 --- a/python/ccxt/async/jubi.py +++ b/python/ccxt/async/jubi.py @@ -12,8 +12,10 @@ def describe(self): 'countries': 'CN', 'rateLimit': 1500, 'version': 'v1', - 'hasCORS': False, - 'hasFetchTickers': True, + 'has': { + 'CORS': False, + 'fetchTickers': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766581-9d397d9a-5edd-11e7-8fb9-5d8236c0e692.jpg', 'api': 'https://www.jubi.com/api', diff --git a/python/ccxt/async/kraken.py b/python/ccxt/async/kraken.py index aa6aa26a92711..413b7fffd496d 100644 --- a/python/ccxt/async/kraken.py +++ b/python/ccxt/async/kraken.py @@ -23,18 +23,8 @@ def describe(self): 'countries': 'US', 'version': '0', 'rateLimit': 3000, - 'hasCORS': False, - # obsolete metainfo interface - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, - 'hasFetchOrder': True, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchMyTrades': True, - 'hasWithdraw': True, - 'hasFetchCurrencies': True, - # new metainfo interface 'has': { + 'CORS': False, 'fetchCurrencies': True, 'fetchTickers': True, 'fetchOHLCV': True, diff --git a/python/ccxt/async/kucoin.py b/python/ccxt/async/kucoin.py index 488d4354b3a9e..ee9f131690204 100644 --- a/python/ccxt/async/kucoin.py +++ b/python/ccxt/async/kucoin.py @@ -20,20 +20,9 @@ def describe(self): 'countries': 'HK', # Hong Kong 'version': 'v1', 'rateLimit': 2000, - 'hasCORS': False, 'userAgent': self.userAgents['chrome'], - # obsolete metainfo interface - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, - 'hasFetchOrder': False, - 'hasFetchOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchMyTrades': False, - 'hasFetchCurrencies': True, - 'hasWithdraw': True, - # new metainfo interface 'has': { + 'CORS': False, 'fetchTickers': True, 'fetchOHLCV': True, # see the method implementation below 'fetchOrder': False, diff --git a/python/ccxt/async/kuna.py b/python/ccxt/async/kuna.py index 223fa935bd94b..d6736c0e91673 100644 --- a/python/ccxt/async/kuna.py +++ b/python/ccxt/async/kuna.py @@ -16,9 +16,11 @@ def describe(self): 'countries': 'UA', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': False, - 'hasFetchTickers': False, - 'hasFetchOHLCV': False, + 'has': { + 'CORS': False, + 'fetchTickers': False, + 'fetchOHLCV': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/31697638-912824fa-b3c1-11e7-8c36-cf9606eb94ac.jpg', 'api': 'https://kuna.io', diff --git a/python/ccxt/async/lakebtc.py b/python/ccxt/async/lakebtc.py index cf86b78dbf9f8..ea83113039030 100644 --- a/python/ccxt/async/lakebtc.py +++ b/python/ccxt/async/lakebtc.py @@ -14,7 +14,9 @@ def describe(self): 'name': 'LakeBTC', 'countries': 'US', 'version': 'api_v2', - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28074120-72b7c38a-6660-11e7-92d9-d9027502281d.jpg', 'api': 'https://api.lakebtc.com', diff --git a/python/ccxt/async/liqui.py b/python/ccxt/async/liqui.py index 55e30f3cbc2bc..54a200496158f 100644 --- a/python/ccxt/async/liqui.py +++ b/python/ccxt/async/liqui.py @@ -29,18 +29,9 @@ def describe(self): 'countries': 'UA', 'rateLimit': 3000, 'version': '3', - 'hasCORS': False, 'userAgent': self.userAgents['chrome'], - # obsolete metainfo interface - 'hasFetchOrder': True, - 'hasFetchOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchTickers': True, - 'hasFetchMyTrades': True, - 'hasWithdraw': True, - # new metainfo interface 'has': { + 'CORS': False, 'fetchOrder': True, 'fetchOrders': 'emulated', 'fetchOpenOrders': True, diff --git a/python/ccxt/async/livecoin.py b/python/ccxt/async/livecoin.py index 673509eca6e5b..9b352efb2904f 100644 --- a/python/ccxt/async/livecoin.py +++ b/python/ccxt/async/livecoin.py @@ -20,12 +20,8 @@ def describe(self): 'name': 'LiveCoin', 'countries': ['US', 'UK', 'RU'], 'rateLimit': 1000, - 'hasCORS': False, - # obsolete metainfo interface - 'hasFetchTickers': True, - 'hasFetchCurrencies': True, - # new metainfo interface 'has': { + 'CORS': False, 'fetchTickers': True, 'fetchCurrencies': True, }, diff --git a/python/ccxt/async/luno.py b/python/ccxt/async/luno.py index 471d40c52a687..e29c922dfc282 100644 --- a/python/ccxt/async/luno.py +++ b/python/ccxt/async/luno.py @@ -14,10 +14,8 @@ def describe(self): 'countries': ['GB', 'SG', 'ZA'], 'rateLimit': 10000, 'version': '1', - 'hasCORS': False, - 'hasFetchTickers': True, - 'hasFetchOrder': True, 'has': { + 'CORS': False, 'fetchTickers': True, 'fetchOrder': True, }, diff --git a/python/ccxt/async/lykke.py b/python/ccxt/async/lykke.py index 2edd0ba817241..081763162c75c 100644 --- a/python/ccxt/async/lykke.py +++ b/python/ccxt/async/lykke.py @@ -13,12 +13,8 @@ def describe(self): 'countries': 'CH', 'version': 'v1', 'rateLimit': 200, - 'hasCORS': False, - # obsolete metainfo interface - 'hasFetchTrades': False, - 'hasFetchOHLCV': False, - # new metainfo interface 'has': { + 'CORS': False, 'fetchOHLCV': False, 'fetchTrades': False, }, diff --git a/python/ccxt/async/mercado.py b/python/ccxt/async/mercado.py index 03d9ba90c7d3a..4b6d91031ac62 100644 --- a/python/ccxt/async/mercado.py +++ b/python/ccxt/async/mercado.py @@ -14,8 +14,10 @@ def describe(self): 'countries': 'BR', # Brazil 'rateLimit': 1000, 'version': 'v3', - 'hasCORS': True, - 'hasWithdraw': True, + 'has': { + 'CORS': True, + 'withdraw': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27837060-e7c58714-60ea-11e7-9192-f05e86adb83f.jpg', 'api': { diff --git a/python/ccxt/async/mixcoins.py b/python/ccxt/async/mixcoins.py index 10deac15c8559..a3dfca6b8bf20 100644 --- a/python/ccxt/async/mixcoins.py +++ b/python/ccxt/async/mixcoins.py @@ -14,7 +14,9 @@ def describe(self): 'countries': ['GB', 'HK'], 'rateLimit': 1500, 'version': 'v1', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30237212-ed29303c-9535-11e7-8af8-fcd381cfa20c.jpg', 'api': 'https://mixcoins.com/api', diff --git a/python/ccxt/async/nova.py b/python/ccxt/async/nova.py index d099984cd10f9..bc4f91dec8a69 100644 --- a/python/ccxt/async/nova.py +++ b/python/ccxt/async/nova.py @@ -14,7 +14,9 @@ def describe(self): 'countries': 'TZ', # Tanzania 'rateLimit': 2000, 'version': 'v2', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30518571-78ca0bca-9b8a-11e7-8840-64b83a4a94b2.jpg', 'api': 'https://novaexchange.com/remote', diff --git a/python/ccxt/async/okcoincny.py b/python/ccxt/async/okcoincny.py index e0806082ebf68..6009829cb5b11 100644 --- a/python/ccxt/async/okcoincny.py +++ b/python/ccxt/async/okcoincny.py @@ -10,7 +10,9 @@ def describe(self): 'id': 'okcoincny', 'name': 'OKCoin CNY', 'countries': 'CN', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766792-8be9157a-5ee5-11e7-926c-6d69b8d3378d.jpg', 'api': { diff --git a/python/ccxt/async/okcoinusd.py b/python/ccxt/async/okcoinusd.py index 8cf0d1a98439f..099aafd9d9112 100644 --- a/python/ccxt/async/okcoinusd.py +++ b/python/ccxt/async/okcoinusd.py @@ -17,27 +17,19 @@ def describe(self): 'id': 'okcoinusd', 'name': 'OKCoin USD', 'countries': ['CN', 'US'], - 'hasCORS': False, 'version': 'v1', 'rateLimit': 1000, # up to 3000 requests per 5 minutes ≈ 600 requests per minute ≈ 10 requests per second ≈ 100 ms - # obsolete metainfo interface - 'hasFetchOHLCV': True, - 'hasFetchOrder': True, - 'hasFetchOrders': False, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasWithdraw': True, - # new metainfo interface 'has': { + 'CORS': False, 'fetchOHLCV': True, 'fetchOrder': True, 'fetchOrders': False, 'fetchOpenOrders': True, 'fetchClosedOrders': True, 'withdraw': True, + 'futureMarkets': False, }, 'extension': '.do', # appended to endpoint URL - 'hasFutureMarkets': False, 'timeframes': { '1m': '1min', '3m': '3min', @@ -194,7 +186,7 @@ async def fetch_markets(self): }, }) result.append(market) - if (self.hasFutureMarkets) and(market['quote'] == 'USDT'): + if (self.has['futureMarkets']) and(market['quote'] == 'USDT'): result.append(self.extend(market, { 'quote': 'USD', 'symbol': market['base'] + '/USD', diff --git a/python/ccxt/async/okex.py b/python/ccxt/async/okex.py index 86cfe2875d79b..9d33913a5a041 100644 --- a/python/ccxt/async/okex.py +++ b/python/ccxt/async/okex.py @@ -10,8 +10,10 @@ def describe(self): 'id': 'okex', 'name': 'OKEX', 'countries': ['CN', 'US'], - 'hasCORS': False, - 'hasFutureMarkets': True, + 'has': { + 'CORS': False, + 'futureMarkets': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/32552768-0d6dd3c6-c4a6-11e7-90f8-c043b64756a7.jpg', 'api': { diff --git a/python/ccxt/async/paymium.py b/python/ccxt/async/paymium.py index d0a1e20dcd555..291333e1db2bb 100644 --- a/python/ccxt/async/paymium.py +++ b/python/ccxt/async/paymium.py @@ -13,7 +13,9 @@ def describe(self): 'countries': ['FR', 'EU'], 'rateLimit': 2000, 'version': 'v1', - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27790564-a945a9d4-5ff9-11e7-9d2d-b635763f2f24.jpg', 'api': 'https://paymium.com/api', diff --git a/python/ccxt/async/poloniex.py b/python/ccxt/async/poloniex.py index 7b5e9ef0d189b..a7eb5e0ffb44a 100644 --- a/python/ccxt/async/poloniex.py +++ b/python/ccxt/async/poloniex.py @@ -20,19 +20,8 @@ def describe(self): 'name': 'Poloniex', 'countries': 'US', 'rateLimit': 1000, # up to 6 calls per second - 'hasCORS': True, - # obsolete metainfo interface - 'hasFetchMyTrades': True, - 'hasFetchOrder': True, - 'hasFetchOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchTickers': True, - 'hasFetchCurrencies': True, - 'hasWithdraw': True, - 'hasFetchOHLCV': True, - # new metainfo interface 'has': { + 'CORS': True, 'fetchOHLCV': True, 'fetchMyTrades': True, 'fetchOrder': 'emulated', diff --git a/python/ccxt/async/qryptos.py b/python/ccxt/async/qryptos.py index af1c67cef4ab1..5a560d509778e 100644 --- a/python/ccxt/async/qryptos.py +++ b/python/ccxt/async/qryptos.py @@ -19,9 +19,9 @@ def describe(self): 'countries': ['CN', 'TW'], 'version': '2', 'rateLimit': 1000, - 'hasFetchTickers': True, - 'hasCORS': False, 'has': { + 'CORS': False, + 'fetchTickers': True, 'fetchOrder': True, 'fetchOrders': True, 'fetchOpenOrders': True, diff --git a/python/ccxt/async/quadrigacx.py b/python/ccxt/async/quadrigacx.py index d91ad6cfbee83..d149e3539cefd 100644 --- a/python/ccxt/async/quadrigacx.py +++ b/python/ccxt/async/quadrigacx.py @@ -22,11 +22,8 @@ def describe(self): 'countries': 'CA', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': True, - # obsolete metainfo interface - 'hasWithdraw': True, - # new metainfo interface 'has': { + 'CORS': True, 'withdraw': True, }, 'urls': { diff --git a/python/ccxt/async/quoinex.py b/python/ccxt/async/quoinex.py index 2422cc77243f2..fd9d6f8d183c2 100644 --- a/python/ccxt/async/quoinex.py +++ b/python/ccxt/async/quoinex.py @@ -12,8 +12,10 @@ def describe(self): 'countries': ['JP', 'SG', 'VN'], 'version': '2', 'rateLimit': 1000, - 'hasFetchTickers': True, - 'hasCORS': False, + 'has': { + 'CORS': False, + 'fetchTickers': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/35047114-0e24ad4a-fbaa-11e7-96a9-69c1a756083b.jpg', 'api': 'https://api.quoine.com', diff --git a/python/ccxt/async/southxchange.py b/python/ccxt/async/southxchange.py index 6f2d1b59aa2f9..e66dbbc4d50c3 100644 --- a/python/ccxt/async/southxchange.py +++ b/python/ccxt/async/southxchange.py @@ -13,9 +13,11 @@ def describe(self): 'name': 'SouthXchange', 'countries': 'AR', # Argentina 'rateLimit': 1000, - 'hasFetchTickers': True, - 'hasCORS': False, - 'hasWithdraw': True, + 'has': { + 'CORS': True, + 'fetchTickers': True, + 'withdraw': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27838912-4f94ec8a-60f6-11e7-9e5d-bbf9bd50a559.jpg', 'api': 'https://www.southxchange.com/api', diff --git a/python/ccxt/async/surbitcoin.py b/python/ccxt/async/surbitcoin.py index f649f35a54b18..89d430337e626 100644 --- a/python/ccxt/async/surbitcoin.py +++ b/python/ccxt/async/surbitcoin.py @@ -10,7 +10,9 @@ def describe(self): 'id': 'surbitcoin', 'name': 'SurBitcoin', 'countries': 'VE', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27991511-f0a50194-6481-11e7-99b5-8f02932424cc.jpg', 'api': { diff --git a/python/ccxt/async/therock.py b/python/ccxt/async/therock.py index 7d969cc495ec3..ee13fd9266478 100644 --- a/python/ccxt/async/therock.py +++ b/python/ccxt/async/therock.py @@ -14,8 +14,10 @@ def describe(self): 'countries': 'MT', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': False, - 'hasFetchTickers': True, + 'has': { + 'CORS': False, + 'fetchTickers': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766869-75057fa2-5ee9-11e7-9a6f-13e641fa4707.jpg', 'api': 'https://api.therocktrading.com', diff --git a/python/ccxt/async/tidex.py b/python/ccxt/async/tidex.py index a3709644e8dbb..758dc4f5e403a 100644 --- a/python/ccxt/async/tidex.py +++ b/python/ccxt/async/tidex.py @@ -12,8 +12,10 @@ def describe(self): 'countries': 'UK', 'rateLimit': 2000, 'version': '3', - # 'hasCORS': False, - # 'hasFetchTickers': True, + 'has': { + # 'CORS': False, + # 'fetchTickers': True + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30781780-03149dc4-a12e-11e7-82bb-313b269d24d4.jpg', 'api': { diff --git a/python/ccxt/async/urdubit.py b/python/ccxt/async/urdubit.py index cd971659a9912..d47e5a58fc90c 100644 --- a/python/ccxt/async/urdubit.py +++ b/python/ccxt/async/urdubit.py @@ -10,7 +10,9 @@ def describe(self): 'id': 'urdubit', 'name': 'UrduBit', 'countries': 'PK', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27991453-156bf3ae-6480-11e7-82eb-7295fe1b5bb4.jpg', 'api': { diff --git a/python/ccxt/async/vaultoro.py b/python/ccxt/async/vaultoro.py index 7ad31104ba735..88de451690447 100644 --- a/python/ccxt/async/vaultoro.py +++ b/python/ccxt/async/vaultoro.py @@ -12,7 +12,9 @@ def describe(self): 'countries': 'CH', 'rateLimit': 1000, 'version': '1', - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766880-f205e870-5ee9-11e7-8fe2-0d5b15880752.jpg', 'api': 'https://api.vaultoro.com', diff --git a/python/ccxt/async/vbtc.py b/python/ccxt/async/vbtc.py index fbcdebbcb1c12..690a4c7278e5d 100644 --- a/python/ccxt/async/vbtc.py +++ b/python/ccxt/async/vbtc.py @@ -10,7 +10,9 @@ def describe(self): 'id': 'vbtc', 'name': 'VBTC', 'countries': 'VN', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27991481-1f53d1d8-6481-11e7-884e-21d17e7939db.jpg', 'api': { diff --git a/python/ccxt/async/virwox.py b/python/ccxt/async/virwox.py index e559987a38919..a1fe57f48ae10 100644 --- a/python/ccxt/async/virwox.py +++ b/python/ccxt/async/virwox.py @@ -13,7 +13,9 @@ def describe(self): 'name': 'VirWoX', 'countries': ['AT', 'EU'], 'rateLimit': 1000, - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766894-6da9d360-5eea-11e7-90aa-41f2711b7405.jpg', 'api': { diff --git a/python/ccxt/async/wex.py b/python/ccxt/async/wex.py index 50aa99bd3d755..77b261b114596 100644 --- a/python/ccxt/async/wex.py +++ b/python/ccxt/async/wex.py @@ -16,8 +16,10 @@ def describe(self): 'name': 'WEX', 'countries': 'NZ', # New Zealand 'version': '3', - 'hasFetchTickers': True, - 'hasCORS': False, + 'has': { + 'CORS': False, + 'fetchTickers': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30652751-d74ec8f8-9e31-11e7-98c5-71469fcef03e.jpg', 'api': { diff --git a/python/ccxt/async/xbtce.py b/python/ccxt/async/xbtce.py index fa0f536b1b09d..6b185c2070df1 100644 --- a/python/ccxt/async/xbtce.py +++ b/python/ccxt/async/xbtce.py @@ -16,10 +16,12 @@ def describe(self): 'countries': 'RU', 'rateLimit': 2000, # responses are cached every 2 seconds 'version': 'v1', - 'hasPublicAPI': False, - 'hasCORS': False, - 'hasFetchTickers': True, - 'hasFetchOHLCV': False, + 'has': { + 'publicAPI': False, + 'CORS': False, + 'fetchTickers': True, + 'fetchOHLCV': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28059414-e235970c-662c-11e7-8c3a-08e31f78684b.jpg', 'api': 'https://cryptottlivewebapi.xbtce.net:8443/api', diff --git a/python/ccxt/async/yobit.py b/python/ccxt/async/yobit.py index a9bc88a647f31..9648c4286772b 100644 --- a/python/ccxt/async/yobit.py +++ b/python/ccxt/async/yobit.py @@ -15,8 +15,10 @@ def describe(self): 'countries': 'RU', 'rateLimit': 3000, # responses are cached every 2 seconds 'version': '3', - 'hasCORS': False, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'withdraw': True + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766910-cdcbfdae-5eea-11e7-9859-03fea873272d.jpg', 'api': { diff --git a/python/ccxt/async/yunbi.py b/python/ccxt/async/yunbi.py index 60d5d642d4f1e..10d43c40409fe 100644 --- a/python/ccxt/async/yunbi.py +++ b/python/ccxt/async/yunbi.py @@ -12,9 +12,11 @@ def describe(self): 'countries': 'CN', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': False, - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, + 'has': { + 'CORS': False, + 'fetchTickers': True, + 'fetchOHLCV': True, + }, 'timeframes': { '1m': '1', '5m': '5', diff --git a/python/ccxt/async/zaif.py b/python/ccxt/async/zaif.py index 0b9eb64e67b6c..c4be1988c0e8a 100644 --- a/python/ccxt/async/zaif.py +++ b/python/ccxt/async/zaif.py @@ -14,10 +14,12 @@ def describe(self): 'countries': 'JP', 'rateLimit': 2000, 'version': '1', - 'hasCORS': False, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'fetchOpenOrders': True, + 'fetchClosedOrders': True, + 'withdraw': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766927-39ca2ada-5eeb-11e7-972f-1b4199518ca6.jpg', 'api': 'https://api.zaif.jp', diff --git a/python/ccxt/async/zb.py b/python/ccxt/async/zb.py index 9463145d4778e..7d12699236b54 100644 --- a/python/ccxt/async/zb.py +++ b/python/ccxt/async/zb.py @@ -15,8 +15,10 @@ def describe(self): 'countries': 'CN', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': False, - 'hasFetchOrder': True, + 'has': { + 'CORS': False, + 'fetchOrder': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/32859187-cd5214f0-ca5e-11e7-967d-96568e2e2bd1.jpg', 'api': { diff --git a/python/ccxt/base/exchange.py b/python/ccxt/base/exchange.py index 1ae6a0cf7f758..2648b552f8271 100644 --- a/python/ccxt/base/exchange.py +++ b/python/ccxt/base/exchange.py @@ -115,26 +115,6 @@ class Exchange(object): markets_by_id = None currencies_by_id = None - hasPublicAPI = True - hasPrivateAPI = True - hasCORS = False - hasFetchTicker = True - hasFetchOrderBook = True - hasFetchTrades = True - hasFetchTickers = False - hasFetchOHLCV = False - hasDeposit = False - hasWithdraw = False - hasFetchBalance = True - hasFetchOrder = False - hasFetchOrders = False - hasFetchOpenOrders = False - hasFetchClosedOrders = False - hasFetchMyTrades = False - hasFetchCurrencies = False - hasCreateOrder = hasPrivateAPI - hasCancelOrder = hasPrivateAPI - requiredCredentials = { 'apiKey': True, 'secret': True, @@ -145,9 +125,12 @@ class Exchange(object): # API method metainfo has = { - 'cancelOrder': hasPrivateAPI, + 'publicAPI': True, + 'privateAPI': True, + 'CORS': True, + 'cancelOrder': True, 'createDepositAddress': False, - 'createOrder': hasPrivateAPI, + 'createOrder': True, 'deposit': False, 'fetchBalance': True, 'fetchClosedOrders': False, diff --git a/python/ccxt/bibox.py b/python/ccxt/bibox.py index 1d07ba84fe37c..8b4308c76f620 100644 --- a/python/ccxt/bibox.py +++ b/python/ccxt/bibox.py @@ -16,16 +16,9 @@ def describe(self): 'name': 'Bibox', 'countries': ['CN', 'US', 'KR'], 'version': 'v1', - 'hasCORS': False, - 'hasPublicAPI': False, - 'hasFetchBalance': True, - 'hasFetchCurrencies': True, - 'hasFetchTickers': True, - 'hasFetchOrders': True, - 'hasFetchMyTrades': True, - 'hasFetchOHLCV': True, - 'hasWithdraw': True, 'has': { + 'CORS': False, + 'publicAPI': False, 'fetchBalance': True, 'fetchCurrencies': True, 'fetchTickers': True, diff --git a/python/ccxt/binance.py b/python/ccxt/binance.py index 4b9ab620cb670..fbd48a0effab8 100644 --- a/python/ccxt/binance.py +++ b/python/ccxt/binance.py @@ -18,18 +18,9 @@ def describe(self): 'name': 'Binance', 'countries': 'JP', # Japan 'rateLimit': 500, - 'hasCORS': False, - # obsolete metainfo interface - 'hasFetchBidsAsks': True, - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, - 'hasFetchMyTrades': True, - 'hasFetchOrder': True, - 'hasFetchOrders': True, - 'hasFetchOpenOrders': True, - 'hasWithdraw': True, # new metainfo interface 'has': { + 'CORS': False, 'fetchBidsAsks': True, 'fetchTickers': True, 'fetchOHLCV': True, diff --git a/python/ccxt/bit2c.py b/python/ccxt/bit2c.py index c0a4014aaf9f7..b789a5f86edd2 100644 --- a/python/ccxt/bit2c.py +++ b/python/ccxt/bit2c.py @@ -12,7 +12,9 @@ def describe(self): 'name': 'Bit2C', 'countries': 'IL', # Israel 'rateLimit': 3000, - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766119-3593220e-5ece-11e7-8b3a-5a041f6bcc3f.jpg', 'api': 'https://www.bit2c.co.il', diff --git a/python/ccxt/bitbay.py b/python/ccxt/bitbay.py index 51b5e909d4e0d..bcbd4c5d7e3a0 100644 --- a/python/ccxt/bitbay.py +++ b/python/ccxt/bitbay.py @@ -27,8 +27,10 @@ def describe(self): 'name': 'BitBay', 'countries': ['PL', 'EU'], # Poland 'rateLimit': 1000, - 'hasCORS': True, - 'hasWithdraw': True, + 'has': { + 'CORS': True, + 'withdraw': True + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766132-978a7bd8-5ece-11e7-9540-bc96d1e9bbb8.jpg', 'www': 'https://bitbay.net', diff --git a/python/ccxt/bitcoincoid.py b/python/ccxt/bitcoincoid.py index 3162369be883b..a772539645251 100644 --- a/python/ccxt/bitcoincoid.py +++ b/python/ccxt/bitcoincoid.py @@ -12,19 +12,8 @@ def describe(self): 'id': 'bitcoincoid', 'name': 'Bitcoin.co.id', 'countries': 'ID', # Indonesia - 'hasCORS': False, - # obsolete metainfo interface - 'hasFetchTickers': False, - 'hasFetchOHLCV': False, - 'hasFetchOrder': True, - 'hasFetchOrders': False, - 'hasFetchClosedOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchMyTrades': False, - 'hasFetchCurrencies': False, - 'hasWithdraw': False, - # new metainfo interface 'has': { + 'CORS': False, 'fetchTickers': False, 'fetchOHLCV': False, 'fetchOrder': True, diff --git a/python/ccxt/bitfinex.py b/python/ccxt/bitfinex.py index 5dcc1c7759bf9..9172de453d773 100644 --- a/python/ccxt/bitfinex.py +++ b/python/ccxt/bitfinex.py @@ -23,17 +23,9 @@ def describe(self): 'countries': 'VG', 'version': 'v1', 'rateLimit': 1500, - 'hasCORS': False, - # old metainfo interface - 'hasFetchOrder': True, - 'hasFetchTickers': True, - 'hasDeposit': True, - 'hasWithdraw': True, - 'hasFetchOHLCV': True, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, # new metainfo interface 'has': { + 'CORS': False, 'fetchOHLCV': True, 'fetchTickers': True, 'fetchOrder': True, diff --git a/python/ccxt/bitfinex2.py b/python/ccxt/bitfinex2.py index 42298913a5201..8309fadc82a78 100644 --- a/python/ccxt/bitfinex2.py +++ b/python/ccxt/bitfinex2.py @@ -15,18 +15,9 @@ def describe(self): 'name': 'Bitfinex v2', 'countries': 'VG', 'version': 'v2', - 'hasCORS': True, - # old metainfo interface - 'hasCreateOrder': False, - 'hasFetchOrder': True, - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, - 'hasWithdraw': True, - 'hasDeposit': False, - 'hasFetchOpenOrders': False, - 'hasFetchClosedOrders': False, # new metainfo interface 'has': { + 'CORS': True, 'createOrder': False, 'fetchOHLCV': True, 'fetchTickers': True, diff --git a/python/ccxt/bitflyer.py b/python/ccxt/bitflyer.py index aa75d1394637d..6931d163887d3 100644 --- a/python/ccxt/bitflyer.py +++ b/python/ccxt/bitflyer.py @@ -12,8 +12,10 @@ def describe(self): 'countries': 'JP', 'version': 'v1', 'rateLimit': 500, - 'hasCORS': False, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'withdraw': True + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28051642-56154182-660e-11e7-9b0d-6042d1e6edd8.jpg', 'api': 'https://api.bitflyer.jp', diff --git a/python/ccxt/bithumb.py b/python/ccxt/bithumb.py index fd0e7b9c607a2..743fcf8997946 100644 --- a/python/ccxt/bithumb.py +++ b/python/ccxt/bithumb.py @@ -14,12 +14,8 @@ def describe(self): 'name': 'Bithumb', 'countries': 'KR', # South Korea 'rateLimit': 500, - 'hasCORS': True, - # obsolete metainfo interface - 'hasFetchTickers': True, - 'hasWithdraw': True, - # new metainfo interface 'has': { + 'CORS': True, 'fetchTickers': True, 'withdraw': True, }, diff --git a/python/ccxt/bitlish.py b/python/ccxt/bitlish.py index b550f2d5b2179..5a1d107173eae 100644 --- a/python/ccxt/bitlish.py +++ b/python/ccxt/bitlish.py @@ -13,10 +13,12 @@ def describe(self): 'countries': ['GB', 'EU', 'RU'], 'rateLimit': 1500, 'version': 'v1', - 'hasCORS': False, - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'fetchTickers': True, + 'fetchOHLCV': True, + 'withdraw': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766275-dcfc6c30-5ed3-11e7-839d-00a846385d0b.jpg', 'api': 'https://bitlish.com/api', diff --git a/python/ccxt/bitmarket.py b/python/ccxt/bitmarket.py index 442f9940ef530..59022080e44f4 100644 --- a/python/ccxt/bitmarket.py +++ b/python/ccxt/bitmarket.py @@ -13,9 +13,11 @@ def describe(self): 'name': 'BitMarket', 'countries': ['PL', 'EU'], 'rateLimit': 1500, - 'hasCORS': False, - 'hasFetchOHLCV': True, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'fetchOHLCV': True, + 'withdraw': True, + }, 'timeframes': { '90m': '90m', '6h': '6h', diff --git a/python/ccxt/bitmex.py b/python/ccxt/bitmex.py index 566734191edc5..a38f84c3a6439 100644 --- a/python/ccxt/bitmex.py +++ b/python/ccxt/bitmex.py @@ -16,9 +16,11 @@ def describe(self): 'version': 'v1', 'userAgent': None, 'rateLimit': 1500, - 'hasCORS': False, - 'hasFetchOHLCV': True, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'fetchOHLCV': True, + 'withdraw': True, + }, 'timeframes': { '1m': '1m', '5m': '5m', diff --git a/python/ccxt/bitso.py b/python/ccxt/bitso.py index 523f34f25eb82..54aff00430fd9 100644 --- a/python/ccxt/bitso.py +++ b/python/ccxt/bitso.py @@ -13,7 +13,9 @@ def describe(self): 'countries': 'MX', # Mexico 'rateLimit': 2000, # 30 requests per minute 'version': 'v3', - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766335-715ce7aa-5ed5-11e7-88a8-173a27bb30fe.jpg', 'api': 'https://api.bitso.com', diff --git a/python/ccxt/bitstamp.py b/python/ccxt/bitstamp.py index ce4a0cf47a91f..aaa7393f26ad9 100644 --- a/python/ccxt/bitstamp.py +++ b/python/ccxt/bitstamp.py @@ -14,12 +14,8 @@ def describe(self): 'countries': 'GB', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': False, - # obsolete metainfo interface - 'hasFetchOrder': True, - 'hasWithdraw': True, - # new metainfo interface 'has': { + 'CORS': True, 'fetchOrder': True, 'withdraw': True, }, diff --git a/python/ccxt/bitstamp1.py b/python/ccxt/bitstamp1.py index c71ac280f6421..d2896324156fe 100644 --- a/python/ccxt/bitstamp1.py +++ b/python/ccxt/bitstamp1.py @@ -14,7 +14,9 @@ def describe(self): 'countries': 'GB', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27786377-8c8ab57e-5fe9-11e7-8ea4-2b05b6bcceec.jpg', 'api': 'https://www.bitstamp.net/api', diff --git a/python/ccxt/bittrex.py b/python/ccxt/bittrex.py index 3738b6c81f6bb..f655e24aecb34 100644 --- a/python/ccxt/bittrex.py +++ b/python/ccxt/bittrex.py @@ -31,19 +31,9 @@ def describe(self): 'version': 'v1.1', 'rateLimit': 1500, 'hasAlreadyAuthenticatedSuccessfully': False, # a workaround for APIKEY_INVALID - 'hasCORS': False, - # obsolete metainfo interface - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, - 'hasFetchOrder': True, - 'hasFetchOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchMyTrades': False, - 'hasFetchCurrencies': True, - 'hasWithdraw': True, # new metainfo interface 'has': { + 'CORS': True, 'fetchTickers': True, 'fetchOHLCV': True, 'fetchOrder': True, diff --git a/python/ccxt/bl3p.py b/python/ccxt/bl3p.py index 71edb8df2f2b4..64ce08f48b793 100644 --- a/python/ccxt/bl3p.py +++ b/python/ccxt/bl3p.py @@ -15,7 +15,9 @@ def describe(self): 'rateLimit': 1000, 'version': '1', 'comment': 'An exchange market by BitonicNL', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28501752-60c21b82-6feb-11e7-818b-055ee6d0e754.jpg', 'api': 'https://api.bl3p.eu', diff --git a/python/ccxt/bleutrade.py b/python/ccxt/bleutrade.py index 3f243a55250d4..0313b3a0181a7 100644 --- a/python/ccxt/bleutrade.py +++ b/python/ccxt/bleutrade.py @@ -17,9 +17,11 @@ def describe(self): 'countries': 'BR', # Brazil 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': True, - 'hasFetchTickers': True, - 'hasFetchOHLCV': False, + 'has': { + 'CORS': True, + 'fetchTickers': True, + 'fetchOHLCV': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30303000-b602dbe6-976d-11e7-956d-36c5049c01e7.jpg', 'api': { diff --git a/python/ccxt/braziliex.py b/python/ccxt/braziliex.py index a2032e66cc868..9ba5e6d9d7dfe 100644 --- a/python/ccxt/braziliex.py +++ b/python/ccxt/braziliex.py @@ -16,11 +16,6 @@ def describe(self): 'name': 'Braziliex', 'countries': 'BR', 'rateLimit': 1000, - # obsolete metainfo interface - 'hasFetchTickers': True, - 'hasFetchOpenOrders': True, - 'hasFetchMyTrades': True, - # new metainfo interface 'has': { 'fetchTickers': True, 'fetchOpenOrders': True, diff --git a/python/ccxt/btcbox.py b/python/ccxt/btcbox.py index d52ae3cb3449e..c404c6ada3b26 100644 --- a/python/ccxt/btcbox.py +++ b/python/ccxt/btcbox.py @@ -13,8 +13,10 @@ def describe(self): 'countries': 'JP', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': False, - 'hasFetchOHLCV': False, + 'has': { + 'CORS': False, + 'fetchOHLCV': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/31275803-4df755a8-aaa1-11e7-9abb-11ec2fad9f2d.jpg', 'api': 'https://www.btcbox.co.jp/api', diff --git a/python/ccxt/btcchina.py b/python/ccxt/btcchina.py index 528c6a23c5177..0a9853ad036c2 100644 --- a/python/ccxt/btcchina.py +++ b/python/ccxt/btcchina.py @@ -14,7 +14,9 @@ def describe(self): 'countries': 'CN', 'rateLimit': 1500, 'version': 'v1', - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766368-465b3286-5ed6-11e7-9a11-0f6467e1d82b.jpg', 'api': { diff --git a/python/ccxt/btcexchange.py b/python/ccxt/btcexchange.py index 94f9d09a5fb56..a81e7f3c12257 100644 --- a/python/ccxt/btcexchange.py +++ b/python/ccxt/btcexchange.py @@ -11,7 +11,9 @@ def describe(self): 'name': 'BTCExchange', 'countries': 'PH', # Philippines 'rateLimit': 1500, - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27993052-4c92911a-64aa-11e7-96d8-ec6ac3435757.jpg', 'api': 'https://www.btcexchange.ph/api', diff --git a/python/ccxt/btcmarkets.py b/python/ccxt/btcmarkets.py index 347bce703cf22..8c645c057dc8e 100644 --- a/python/ccxt/btcmarkets.py +++ b/python/ccxt/btcmarkets.py @@ -16,13 +16,8 @@ def describe(self): 'name': 'BTC Markets', 'countries': 'AU', # Australia 'rateLimit': 1000, # market data cached for 1 second(trades cached for 2 seconds) - 'hasCORS': False, - 'hasFetchOrder': True, - 'hasFetchOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchMyTrades': True, 'has': { + 'CORS': False, 'fetchOrder': True, 'fetchOrders': True, 'fetchClosedOrders': 'emulated', diff --git a/python/ccxt/btctradeua.py b/python/ccxt/btctradeua.py index aabd600acfe47..d68cd937e559b 100644 --- a/python/ccxt/btctradeua.py +++ b/python/ccxt/btctradeua.py @@ -12,7 +12,9 @@ def describe(self): 'name': 'BTC Trade UA', 'countries': 'UA', # Ukraine, 'rateLimit': 3000, - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27941483-79fc7350-62d9-11e7-9f61-ac47f28fcd96.jpg', 'api': 'https://btc-trade.com.ua/api', diff --git a/python/ccxt/btcturk.py b/python/ccxt/btcturk.py index c41d4dad4b611..a1adefbe5b196 100644 --- a/python/ccxt/btcturk.py +++ b/python/ccxt/btcturk.py @@ -14,9 +14,11 @@ def describe(self): 'name': 'BTCTurk', 'countries': 'TR', # Turkey 'rateLimit': 1000, - 'hasCORS': True, - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, + 'has': { + 'CORS': True, + 'fetchTickers': True, + 'fetchOHLCV': True, + }, 'timeframes': { '1d': '1d', }, diff --git a/python/ccxt/btcx.py b/python/ccxt/btcx.py index c683b45e9facc..4d732304f9a46 100644 --- a/python/ccxt/btcx.py +++ b/python/ccxt/btcx.py @@ -14,7 +14,9 @@ def describe(self): 'countries': ['IS', 'US', 'EU'], 'rateLimit': 1500, # support in english is very poor, unable to tell rate limits 'version': 'v1', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766385-9fdcc98c-5ed6-11e7-8f14-66d5e5cd47e6.jpg', 'api': 'https://btc-x.is/api', diff --git a/python/ccxt/bter.py b/python/ccxt/bter.py index 06ff31ba4ac29..2af4aed295d53 100644 --- a/python/ccxt/bter.py +++ b/python/ccxt/bter.py @@ -13,11 +13,6 @@ def describe(self): 'name': 'Bter', 'countries': ['VG', 'CN'], # British Virgin Islands, China 'version': '2', - # obsolete metainfo interface - 'hasCORS': False, - 'hasFetchTickers': True, - 'hasWithdraw': True, - # new metainfo interface 'has': { 'CORS': False, 'fetchTickers': True, diff --git a/python/ccxt/bxinth.py b/python/ccxt/bxinth.py index 0f88e13701b13..66777ffdac6ac 100644 --- a/python/ccxt/bxinth.py +++ b/python/ccxt/bxinth.py @@ -12,8 +12,10 @@ def describe(self): 'name': 'BX.in.th', 'countries': 'TH', # Thailand 'rateLimit': 1500, - 'hasCORS': False, - 'hasFetchTickers': True, + 'has': { + 'CORS': False, + 'fetchTickers': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766412-567b1eb4-5ed7-11e7-94a8-ff6a3884f6c5.jpg', 'api': 'https://bx.in.th/api', diff --git a/python/ccxt/ccex.py b/python/ccxt/ccex.py index 5d67b1c763ba1..d6b04e269356b 100644 --- a/python/ccxt/ccex.py +++ b/python/ccxt/ccex.py @@ -13,8 +13,10 @@ def describe(self): 'name': 'C-CEX', 'countries': ['DE', 'EU'], 'rateLimit': 1500, - 'hasCORS': False, - 'hasFetchTickers': True, + 'has': { + 'CORS': False, + 'fetchTickers': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766433-16881f90-5ed8-11e7-92f8-3d92cc747a6c.jpg', 'api': { diff --git a/python/ccxt/cex.py b/python/ccxt/cex.py index 805f5f9cbf901..f860990e92a55 100644 --- a/python/ccxt/cex.py +++ b/python/ccxt/cex.py @@ -15,10 +15,12 @@ def describe(self): 'name': 'CEX.IO', 'countries': ['GB', 'EU', 'CY', 'RU'], 'rateLimit': 1500, - 'hasCORS': True, - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, - 'hasFetchOpenOrders': True, + 'has': { + 'CORS': True, + 'fetchTickers': True, + 'fetchOHLCV': True, + 'fetchOpenOrders': True, + }, 'timeframes': { '1m': '1m', }, diff --git a/python/ccxt/chbtc.py b/python/ccxt/chbtc.py index 32b4830bc7087..f3ef942ba6b99 100644 --- a/python/ccxt/chbtc.py +++ b/python/ccxt/chbtc.py @@ -13,8 +13,10 @@ def describe(self): 'countries': 'CN', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': False, - 'hasFetchOrder': True, + 'has': { + 'CORS': False, + 'fetchOrder': True + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28555659-f0040dc2-7109-11e7-9d99-688a438bf9f4.jpg', 'api': { diff --git a/python/ccxt/chilebit.py b/python/ccxt/chilebit.py index 8f7756844bf0d..778912535b6d7 100644 --- a/python/ccxt/chilebit.py +++ b/python/ccxt/chilebit.py @@ -10,7 +10,9 @@ def describe(self): 'id': 'chilebit', 'name': 'ChileBit', 'countries': 'CL', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27991414-1298f0d8-647f-11e7-9c40-d56409266336.jpg', 'api': { diff --git a/python/ccxt/coincheck.py b/python/ccxt/coincheck.py index 65e7b45a2d655..dd6ce47f79fa7 100644 --- a/python/ccxt/coincheck.py +++ b/python/ccxt/coincheck.py @@ -13,7 +13,9 @@ def describe(self): 'name': 'coincheck', 'countries': ['JP', 'ID'], 'rateLimit': 1500, - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766464-3b5c3c74-5ed9-11e7-840e-31b32968e1da.jpg', 'api': 'https://coincheck.com/api', diff --git a/python/ccxt/coinexchange.py b/python/ccxt/coinexchange.py index c3875b6fd3df7..28ab7ed110ac8 100644 --- a/python/ccxt/coinexchange.py +++ b/python/ccxt/coinexchange.py @@ -13,13 +13,9 @@ def describe(self): 'name': 'CoinExchange', 'countries': ['IN', 'JP', 'KR', 'VN', 'US'], 'rateLimit': 1000, - # obsolete metainfo interface - 'hasPrivateAPI': False, - 'hasFetchTrades': False, - 'hasFetchCurrencies': True, - 'hasFetchTickers': True, # new metainfo interface 'has': { + 'privateAPI': False, 'fetchTrades': False, 'fetchCurrencies': True, 'fetchTickers': True, diff --git a/python/ccxt/coinfloor.py b/python/ccxt/coinfloor.py index f36aefc40f1b5..505476821605e 100644 --- a/python/ccxt/coinfloor.py +++ b/python/ccxt/coinfloor.py @@ -13,7 +13,9 @@ def describe(self): 'name': 'coinfloor', 'rateLimit': 1000, 'countries': 'UK', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28246081-623fc164-6a1c-11e7-913f-bac0d5576c90.jpg', 'api': 'https://webapi.coinfloor.co.uk:8090/bist', diff --git a/python/ccxt/coingi.py b/python/ccxt/coingi.py index 82d77995c2069..195a7aa6600d7 100644 --- a/python/ccxt/coingi.py +++ b/python/ccxt/coingi.py @@ -22,8 +22,10 @@ def describe(self): 'name': 'Coingi', 'rateLimit': 1000, 'countries': ['PA', 'BG', 'CN', 'US'], # Panama, Bulgaria, China, US - 'hasFetchTickers': True, - 'hasCORS': False, + 'has': { + 'CORS': False, + 'fetchTickers': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28619707-5c9232a8-7212-11e7-86d6-98fe5d15cc6e.jpg', 'api': { diff --git a/python/ccxt/coinmarketcap.py b/python/ccxt/coinmarketcap.py index e9b4515e342a9..6c5c1835fc34f 100644 --- a/python/ccxt/coinmarketcap.py +++ b/python/ccxt/coinmarketcap.py @@ -14,16 +14,16 @@ def describe(self): 'rateLimit': 10000, 'version': 'v1', 'countries': 'US', - 'hasCORS': True, - 'hasPrivateAPI': False, - 'hasCreateOrder': False, - 'hasCancelOrder': False, - 'hasFetchBalance': False, - 'hasFetchOrderBook': False, - 'hasFetchTrades': False, - 'hasFetchTickers': True, - 'hasFetchCurrencies': True, 'has': { + 'CORS': True, + 'privateAPI': False, + 'createOrder': False, + 'cancelOrder': False, + 'fetchBalance': False, + 'fetchOrderBook': False, + 'fetchTrades': False, + 'fetchTickers': True, + 'fetchCurrencies': True, 'fetchCurrencies': True, }, 'urls': { diff --git a/python/ccxt/coinmate.py b/python/ccxt/coinmate.py index 855a62cd5e7a7..64bcaf8e8d128 100644 --- a/python/ccxt/coinmate.py +++ b/python/ccxt/coinmate.py @@ -12,7 +12,9 @@ def describe(self): 'name': 'CoinMate', 'countries': ['GB', 'CZ', 'EU'], # UK, Czech Republic 'rateLimit': 1000, - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27811229-c1efb510-606c-11e7-9a36-84ba2ce412d8.jpg', 'api': 'https://coinmate.io/api', diff --git a/python/ccxt/coinsecure.py b/python/ccxt/coinsecure.py index 375d23abd7c68..1c8a93dbbdbe0 100644 --- a/python/ccxt/coinsecure.py +++ b/python/ccxt/coinsecure.py @@ -14,7 +14,9 @@ def describe(self): 'countries': 'IN', # India 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766472-9cbd200a-5ed9-11e7-9551-2267ad7bac08.jpg', 'api': 'https://api.coinsecure.in', diff --git a/python/ccxt/coinspot.py b/python/ccxt/coinspot.py index c9379d6c61f47..a45784e9c02df 100644 --- a/python/ccxt/coinspot.py +++ b/python/ccxt/coinspot.py @@ -14,7 +14,9 @@ def describe(self): 'name': 'CoinSpot', 'countries': 'AU', # Australia 'rateLimit': 1000, - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28208429-3cacdf9a-6896-11e7-854e-4c79a772a30f.jpg', 'api': { diff --git a/python/ccxt/cryptopia.py b/python/ccxt/cryptopia.py index b26de6c6f175d..cf848de32fba4 100644 --- a/python/ccxt/cryptopia.py +++ b/python/ccxt/cryptopia.py @@ -18,19 +18,8 @@ def describe(self): 'name': 'Cryptopia', 'rateLimit': 1500, 'countries': 'NZ', # New Zealand - 'hasCORS': False, - # obsolete metainfo interface - 'hasFetchTickers': True, - 'hasFetchOrder': True, - 'hasFetchOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchMyTrades': True, - 'hasFetchCurrencies': True, - 'hasDeposit': True, - 'hasWithdraw': True, - # new metainfo interface 'has': { + 'CORS': False, 'fetchTickers': True, 'fetchOrder': 'emulated', 'fetchOrders': 'emulated', diff --git a/python/ccxt/dsx.py b/python/ccxt/dsx.py index 65b3a617d9d9d..5746371619a47 100644 --- a/python/ccxt/dsx.py +++ b/python/ccxt/dsx.py @@ -12,13 +12,15 @@ def describe(self): 'name': 'DSX', 'countries': 'UK', 'rateLimit': 1500, - 'hasCORS': False, - 'hasFetchOrder': True, - 'hasFetchOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchTickers': True, - 'hasFetchMyTrades': True, + 'has': { + 'CORS': False, + 'fetchOrder': True, + 'fetchOrders': True, + 'fetchOpenOrders': True, + 'fetchClosedOrders': True, + 'fetchTickers': True, + 'fetchMyTrades': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27990275-1413158a-645a-11e7-931c-94717f7510e3.jpg', 'api': { diff --git a/python/ccxt/exmo.py b/python/ccxt/exmo.py index c85cccbe580e0..2c89939791855 100644 --- a/python/ccxt/exmo.py +++ b/python/ccxt/exmo.py @@ -14,9 +14,11 @@ def describe(self): 'countries': ['ES', 'RU'], # Spain, Russia 'rateLimit': 1000, # once every 350 ms ≈ 180 requests per minute ≈ 3 requests per second 'version': 'v1', - 'hasCORS': False, - 'hasFetchTickers': True, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'fetchTickers': True, + 'withdraw': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766491-1b0ea956-5eda-11e7-9225-40d67b481b8d.jpg', 'api': 'https://api.exmo.com', diff --git a/python/ccxt/flowbtc.py b/python/ccxt/flowbtc.py index f4cd732737f43..c450121b2a4e6 100644 --- a/python/ccxt/flowbtc.py +++ b/python/ccxt/flowbtc.py @@ -13,7 +13,9 @@ def describe(self): 'countries': 'BR', # Brazil 'version': 'v1', 'rateLimit': 1000, - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28162465-cd815d4c-67cf-11e7-8e57-438bea0523a2.jpg', 'api': 'https://api.flowbtc.com:8400/ajax', diff --git a/python/ccxt/foxbit.py b/python/ccxt/foxbit.py index 26c140cd22e0b..32207d4877c99 100644 --- a/python/ccxt/foxbit.py +++ b/python/ccxt/foxbit.py @@ -11,7 +11,9 @@ def describe(self): 'id': 'foxbit', 'name': 'FoxBit', 'countries': 'BR', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'rateLimit': 1000, 'version': 'v1', 'urls': { diff --git a/python/ccxt/fybse.py b/python/ccxt/fybse.py index 2a533594f5616..57b882e60f298 100644 --- a/python/ccxt/fybse.py +++ b/python/ccxt/fybse.py @@ -12,7 +12,9 @@ def describe(self): 'id': 'fybse', 'name': 'FYB-SE', 'countries': 'SE', # Sweden - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'rateLimit': 1500, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766512-31019772-5edb-11e7-8241-2e675e6797f1.jpg', diff --git a/python/ccxt/fybsg.py b/python/ccxt/fybsg.py index a8da77169859e..017d7ea7c8fb1 100644 --- a/python/ccxt/fybsg.py +++ b/python/ccxt/fybsg.py @@ -10,7 +10,9 @@ def describe(self): 'id': 'fybsg', 'name': 'FYB-SG', 'countries': 'SG', # Singapore - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766513-3364d56a-5edb-11e7-9e6b-d5898bb89c81.jpg', 'api': 'https://www.fybsg.com/api/SGD', diff --git a/python/ccxt/gatecoin.py b/python/ccxt/gatecoin.py index 7baca57aa64d4..38765fff08379 100644 --- a/python/ccxt/gatecoin.py +++ b/python/ccxt/gatecoin.py @@ -15,9 +15,11 @@ def describe(self): 'rateLimit': 2000, 'countries': 'HK', # Hong Kong 'comment': 'a regulated/licensed exchange', - 'hasCORS': False, - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, + 'has': { + 'CORS': False, + 'fetchTickers': True, + 'fetchOHLCV': True, + }, 'timeframes': { '1m': '1m', '15m': '15m', diff --git a/python/ccxt/gdax.py b/python/ccxt/gdax.py index a237fa403e2ee..2e37d15e745cf 100644 --- a/python/ccxt/gdax.py +++ b/python/ccxt/gdax.py @@ -20,16 +20,6 @@ def describe(self): 'countries': 'US', 'rateLimit': 1000, 'userAgent': self.userAgents['chrome'], - # obsolete metainfo interface - 'hasCORS': True, - 'hasFetchOHLCV': True, - 'hasDeposit': True, - 'hasWithdraw': True, - 'hasFetchOrder': True, - 'hasFetchOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - # new metainfo interface 'has': { 'CORS': True, 'fetchOHLCV': True, diff --git a/python/ccxt/gemini.py b/python/ccxt/gemini.py index e67513b65deb0..a74a0b193567c 100644 --- a/python/ccxt/gemini.py +++ b/python/ccxt/gemini.py @@ -15,10 +15,6 @@ def describe(self): 'countries': 'US', 'rateLimit': 1500, # 200 for private API 'version': 'v1', - # obsolete metainfo interface - 'hasCORS': False, - 'hasWithdraw': True, - # new metainfo interface 'has': { 'CORS': False, 'withdraw': True, diff --git a/python/ccxt/hitbtc.py b/python/ccxt/hitbtc.py index 99f06c1da6d62..3c7880d1d6f30 100644 --- a/python/ccxt/hitbtc.py +++ b/python/ccxt/hitbtc.py @@ -16,12 +16,13 @@ def describe(self): 'countries': 'UK', 'rateLimit': 1500, 'version': '1', - 'hasCORS': False, - 'hasFetchTickers': True, - 'hasFetchOrder': True, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'fetchOrder': True, + 'fetchOpenOrders': True, + 'fetchClosedOrders': True, + 'withdraw': True + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766555-8eaec20e-5edc-11e7-9c5b-6dc69fc42f5e.jpg', 'api': 'http://api.hitbtc.com', diff --git a/python/ccxt/hitbtc2.py b/python/ccxt/hitbtc2.py index b04c17c4ba3f0..e101abf282601 100644 --- a/python/ccxt/hitbtc2.py +++ b/python/ccxt/hitbtc2.py @@ -19,19 +19,8 @@ def describe(self): 'countries': 'UK', 'rateLimit': 1500, 'version': '2', - 'hasCORS': True, - # older metainfo interface - 'hasFetchOHLCV': True, - 'hasFetchTickers': True, - 'hasFetchOrder': True, - 'hasFetchOrders': False, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchMyTrades': True, - 'hasWithdraw': True, - 'hasFetchCurrencies': True, - # new metainfo interface 'has': { + 'CORS': True, 'fetchCurrencies': True, 'fetchOHLCV': True, 'fetchTickers': True, diff --git a/python/ccxt/huobi.py b/python/ccxt/huobi.py index 7197138bbe1d9..07d2399d532f7 100644 --- a/python/ccxt/huobi.py +++ b/python/ccxt/huobi.py @@ -13,8 +13,10 @@ def describe(self): 'countries': 'CN', 'rateLimit': 2000, 'version': 'v3', - 'hasCORS': False, - 'hasFetchOHLCV': True, + 'has': { + 'CORS': False, + 'fetchOHLCV': True, + }, 'timeframes': { '1m': '001', '5m': '005', diff --git a/python/ccxt/huobicny.py b/python/ccxt/huobicny.py index be6def90ac3ec..1ee0f28f65a79 100644 --- a/python/ccxt/huobicny.py +++ b/python/ccxt/huobicny.py @@ -10,7 +10,9 @@ def describe(self): 'id': 'huobicny', 'name': 'Huobi CNY', 'hostname': 'be.huobi.com', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766569-15aa7b9a-5edd-11e7-9e7f-44791f4ee49c.jpg', 'api': 'https://be.huobi.com', diff --git a/python/ccxt/huobipro.py b/python/ccxt/huobipro.py index ffe43f1fecaf2..6b565e55b212d 100644 --- a/python/ccxt/huobipro.py +++ b/python/ccxt/huobipro.py @@ -19,13 +19,8 @@ def describe(self): 'accounts': None, 'accountsById': None, 'hostname': 'api.huobi.pro', - 'hasCORS': False, - # obsolete metainfo structure - 'hasFetchOHLCV': True, - 'hasFetchOrders': True, - 'hasFetchOpenOrders': True, - # new metainfo structure 'has': { + 'CORS': False, 'fetchOHCLV': True, 'fetchOrders': True, 'fetchOpenOrders': True, diff --git a/python/ccxt/independentreserve.py b/python/ccxt/independentreserve.py index be9c92508d08d..98dc67af278c9 100644 --- a/python/ccxt/independentreserve.py +++ b/python/ccxt/independentreserve.py @@ -11,7 +11,9 @@ def describe(self): 'name': 'Independent Reserve', 'countries': ['AU', 'NZ'], # Australia, New Zealand 'rateLimit': 1000, - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30521662-cf3f477c-9bcb-11e7-89bc-d1ac85012eda.jpg', 'api': { diff --git a/python/ccxt/itbit.py b/python/ccxt/itbit.py index cd9f865c44089..a84736f672755 100644 --- a/python/ccxt/itbit.py +++ b/python/ccxt/itbit.py @@ -14,7 +14,9 @@ def describe(self): 'countries': 'US', 'rateLimit': 2000, 'version': 'v1', - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27822159-66153620-60ad-11e7-89e7-005f6d7f3de0.jpg', 'api': 'https://api.itbit.com', diff --git a/python/ccxt/jubi.py b/python/ccxt/jubi.py index 7d0be86b0df49..f793cc0dc0e59 100644 --- a/python/ccxt/jubi.py +++ b/python/ccxt/jubi.py @@ -12,8 +12,10 @@ def describe(self): 'countries': 'CN', 'rateLimit': 1500, 'version': 'v1', - 'hasCORS': False, - 'hasFetchTickers': True, + 'has': { + 'CORS': False, + 'fetchTickers': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766581-9d397d9a-5edd-11e7-8fb9-5d8236c0e692.jpg', 'api': 'https://www.jubi.com/api', diff --git a/python/ccxt/kraken.py b/python/ccxt/kraken.py index e4e91e77cbfe6..ea25272ffd309 100644 --- a/python/ccxt/kraken.py +++ b/python/ccxt/kraken.py @@ -23,18 +23,8 @@ def describe(self): 'countries': 'US', 'version': '0', 'rateLimit': 3000, - 'hasCORS': False, - # obsolete metainfo interface - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, - 'hasFetchOrder': True, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchMyTrades': True, - 'hasWithdraw': True, - 'hasFetchCurrencies': True, - # new metainfo interface 'has': { + 'CORS': False, 'fetchCurrencies': True, 'fetchTickers': True, 'fetchOHLCV': True, diff --git a/python/ccxt/kucoin.py b/python/ccxt/kucoin.py index cbd291511a844..72848b80da265 100644 --- a/python/ccxt/kucoin.py +++ b/python/ccxt/kucoin.py @@ -20,20 +20,9 @@ def describe(self): 'countries': 'HK', # Hong Kong 'version': 'v1', 'rateLimit': 2000, - 'hasCORS': False, 'userAgent': self.userAgents['chrome'], - # obsolete metainfo interface - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, - 'hasFetchOrder': False, - 'hasFetchOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchMyTrades': False, - 'hasFetchCurrencies': True, - 'hasWithdraw': True, - # new metainfo interface 'has': { + 'CORS': False, 'fetchTickers': True, 'fetchOHLCV': True, # see the method implementation below 'fetchOrder': False, diff --git a/python/ccxt/kuna.py b/python/ccxt/kuna.py index 20546f80a8c94..46f90f2982b12 100644 --- a/python/ccxt/kuna.py +++ b/python/ccxt/kuna.py @@ -16,9 +16,11 @@ def describe(self): 'countries': 'UA', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': False, - 'hasFetchTickers': False, - 'hasFetchOHLCV': False, + 'has': { + 'CORS': False, + 'fetchTickers': False, + 'fetchOHLCV': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/31697638-912824fa-b3c1-11e7-8c36-cf9606eb94ac.jpg', 'api': 'https://kuna.io', diff --git a/python/ccxt/lakebtc.py b/python/ccxt/lakebtc.py index 5f4e31048523e..96bd26e8dd732 100644 --- a/python/ccxt/lakebtc.py +++ b/python/ccxt/lakebtc.py @@ -14,7 +14,9 @@ def describe(self): 'name': 'LakeBTC', 'countries': 'US', 'version': 'api_v2', - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28074120-72b7c38a-6660-11e7-92d9-d9027502281d.jpg', 'api': 'https://api.lakebtc.com', diff --git a/python/ccxt/liqui.py b/python/ccxt/liqui.py index c064e916197d3..47605c363cde7 100644 --- a/python/ccxt/liqui.py +++ b/python/ccxt/liqui.py @@ -29,18 +29,9 @@ def describe(self): 'countries': 'UA', 'rateLimit': 3000, 'version': '3', - 'hasCORS': False, 'userAgent': self.userAgents['chrome'], - # obsolete metainfo interface - 'hasFetchOrder': True, - 'hasFetchOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchTickers': True, - 'hasFetchMyTrades': True, - 'hasWithdraw': True, - # new metainfo interface 'has': { + 'CORS': False, 'fetchOrder': True, 'fetchOrders': 'emulated', 'fetchOpenOrders': True, diff --git a/python/ccxt/livecoin.py b/python/ccxt/livecoin.py index 2c4aab03af76e..d2c76a97e0654 100644 --- a/python/ccxt/livecoin.py +++ b/python/ccxt/livecoin.py @@ -20,12 +20,8 @@ def describe(self): 'name': 'LiveCoin', 'countries': ['US', 'UK', 'RU'], 'rateLimit': 1000, - 'hasCORS': False, - # obsolete metainfo interface - 'hasFetchTickers': True, - 'hasFetchCurrencies': True, - # new metainfo interface 'has': { + 'CORS': False, 'fetchTickers': True, 'fetchCurrencies': True, }, diff --git a/python/ccxt/luno.py b/python/ccxt/luno.py index d6d11efe52fcd..270293a82e09d 100644 --- a/python/ccxt/luno.py +++ b/python/ccxt/luno.py @@ -14,10 +14,8 @@ def describe(self): 'countries': ['GB', 'SG', 'ZA'], 'rateLimit': 10000, 'version': '1', - 'hasCORS': False, - 'hasFetchTickers': True, - 'hasFetchOrder': True, 'has': { + 'CORS': False, 'fetchTickers': True, 'fetchOrder': True, }, diff --git a/python/ccxt/lykke.py b/python/ccxt/lykke.py index ac129435b772d..89d6f5ec490d3 100644 --- a/python/ccxt/lykke.py +++ b/python/ccxt/lykke.py @@ -13,12 +13,8 @@ def describe(self): 'countries': 'CH', 'version': 'v1', 'rateLimit': 200, - 'hasCORS': False, - # obsolete metainfo interface - 'hasFetchTrades': False, - 'hasFetchOHLCV': False, - # new metainfo interface 'has': { + 'CORS': False, 'fetchOHLCV': False, 'fetchTrades': False, }, diff --git a/python/ccxt/mercado.py b/python/ccxt/mercado.py index da84d9bad8ad1..479fb954e87a8 100644 --- a/python/ccxt/mercado.py +++ b/python/ccxt/mercado.py @@ -14,8 +14,10 @@ def describe(self): 'countries': 'BR', # Brazil 'rateLimit': 1000, 'version': 'v3', - 'hasCORS': True, - 'hasWithdraw': True, + 'has': { + 'CORS': True, + 'withdraw': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27837060-e7c58714-60ea-11e7-9192-f05e86adb83f.jpg', 'api': { diff --git a/python/ccxt/mixcoins.py b/python/ccxt/mixcoins.py index c5df9598c4fbb..708075a985f13 100644 --- a/python/ccxt/mixcoins.py +++ b/python/ccxt/mixcoins.py @@ -14,7 +14,9 @@ def describe(self): 'countries': ['GB', 'HK'], 'rateLimit': 1500, 'version': 'v1', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30237212-ed29303c-9535-11e7-8af8-fcd381cfa20c.jpg', 'api': 'https://mixcoins.com/api', diff --git a/python/ccxt/nova.py b/python/ccxt/nova.py index ef5118e6c7008..06dc96370b2d0 100644 --- a/python/ccxt/nova.py +++ b/python/ccxt/nova.py @@ -14,7 +14,9 @@ def describe(self): 'countries': 'TZ', # Tanzania 'rateLimit': 2000, 'version': 'v2', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30518571-78ca0bca-9b8a-11e7-8840-64b83a4a94b2.jpg', 'api': 'https://novaexchange.com/remote', diff --git a/python/ccxt/okcoincny.py b/python/ccxt/okcoincny.py index bb275fa426a3a..1d22b87a6be01 100644 --- a/python/ccxt/okcoincny.py +++ b/python/ccxt/okcoincny.py @@ -10,7 +10,9 @@ def describe(self): 'id': 'okcoincny', 'name': 'OKCoin CNY', 'countries': 'CN', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766792-8be9157a-5ee5-11e7-926c-6d69b8d3378d.jpg', 'api': { diff --git a/python/ccxt/okcoinusd.py b/python/ccxt/okcoinusd.py index d8d31773bb523..2a257f10597a5 100644 --- a/python/ccxt/okcoinusd.py +++ b/python/ccxt/okcoinusd.py @@ -17,27 +17,19 @@ def describe(self): 'id': 'okcoinusd', 'name': 'OKCoin USD', 'countries': ['CN', 'US'], - 'hasCORS': False, 'version': 'v1', 'rateLimit': 1000, # up to 3000 requests per 5 minutes ≈ 600 requests per minute ≈ 10 requests per second ≈ 100 ms - # obsolete metainfo interface - 'hasFetchOHLCV': True, - 'hasFetchOrder': True, - 'hasFetchOrders': False, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasWithdraw': True, - # new metainfo interface 'has': { + 'CORS': False, 'fetchOHLCV': True, 'fetchOrder': True, 'fetchOrders': False, 'fetchOpenOrders': True, 'fetchClosedOrders': True, 'withdraw': True, + 'futureMarkets': False, }, 'extension': '.do', # appended to endpoint URL - 'hasFutureMarkets': False, 'timeframes': { '1m': '1min', '3m': '3min', @@ -194,7 +186,7 @@ def fetch_markets(self): }, }) result.append(market) - if (self.hasFutureMarkets) and(market['quote'] == 'USDT'): + if (self.has['futureMarkets']) and(market['quote'] == 'USDT'): result.append(self.extend(market, { 'quote': 'USD', 'symbol': market['base'] + '/USD', diff --git a/python/ccxt/okex.py b/python/ccxt/okex.py index 9be0b62250095..2055ab4905c88 100644 --- a/python/ccxt/okex.py +++ b/python/ccxt/okex.py @@ -10,8 +10,10 @@ def describe(self): 'id': 'okex', 'name': 'OKEX', 'countries': ['CN', 'US'], - 'hasCORS': False, - 'hasFutureMarkets': True, + 'has': { + 'CORS': False, + 'futureMarkets': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/32552768-0d6dd3c6-c4a6-11e7-90f8-c043b64756a7.jpg', 'api': { diff --git a/python/ccxt/paymium.py b/python/ccxt/paymium.py index 3038d2c703dc0..07c8aa487dce3 100644 --- a/python/ccxt/paymium.py +++ b/python/ccxt/paymium.py @@ -13,7 +13,9 @@ def describe(self): 'countries': ['FR', 'EU'], 'rateLimit': 2000, 'version': 'v1', - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27790564-a945a9d4-5ff9-11e7-9d2d-b635763f2f24.jpg', 'api': 'https://paymium.com/api', diff --git a/python/ccxt/poloniex.py b/python/ccxt/poloniex.py index 17b00d142b24f..bd4773827f5fb 100644 --- a/python/ccxt/poloniex.py +++ b/python/ccxt/poloniex.py @@ -20,19 +20,8 @@ def describe(self): 'name': 'Poloniex', 'countries': 'US', 'rateLimit': 1000, # up to 6 calls per second - 'hasCORS': True, - # obsolete metainfo interface - 'hasFetchMyTrades': True, - 'hasFetchOrder': True, - 'hasFetchOrders': True, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasFetchTickers': True, - 'hasFetchCurrencies': True, - 'hasWithdraw': True, - 'hasFetchOHLCV': True, - # new metainfo interface 'has': { + 'CORS': True, 'fetchOHLCV': True, 'fetchMyTrades': True, 'fetchOrder': 'emulated', diff --git a/python/ccxt/qryptos.py b/python/ccxt/qryptos.py index 432ab4a2b8224..1fba4ac90f61b 100644 --- a/python/ccxt/qryptos.py +++ b/python/ccxt/qryptos.py @@ -19,9 +19,9 @@ def describe(self): 'countries': ['CN', 'TW'], 'version': '2', 'rateLimit': 1000, - 'hasFetchTickers': True, - 'hasCORS': False, 'has': { + 'CORS': False, + 'fetchTickers': True, 'fetchOrder': True, 'fetchOrders': True, 'fetchOpenOrders': True, diff --git a/python/ccxt/quadrigacx.py b/python/ccxt/quadrigacx.py index f7334a22dd742..1f13f73a6bd40 100644 --- a/python/ccxt/quadrigacx.py +++ b/python/ccxt/quadrigacx.py @@ -22,11 +22,8 @@ def describe(self): 'countries': 'CA', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': True, - # obsolete metainfo interface - 'hasWithdraw': True, - # new metainfo interface 'has': { + 'CORS': True, 'withdraw': True, }, 'urls': { diff --git a/python/ccxt/quoinex.py b/python/ccxt/quoinex.py index e90831268db9f..2084557f684f0 100644 --- a/python/ccxt/quoinex.py +++ b/python/ccxt/quoinex.py @@ -12,8 +12,10 @@ def describe(self): 'countries': ['JP', 'SG', 'VN'], 'version': '2', 'rateLimit': 1000, - 'hasFetchTickers': True, - 'hasCORS': False, + 'has': { + 'CORS': False, + 'fetchTickers': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/35047114-0e24ad4a-fbaa-11e7-96a9-69c1a756083b.jpg', 'api': 'https://api.quoine.com', diff --git a/python/ccxt/southxchange.py b/python/ccxt/southxchange.py index f73f676d14dd0..0bc78626cd08e 100644 --- a/python/ccxt/southxchange.py +++ b/python/ccxt/southxchange.py @@ -13,9 +13,11 @@ def describe(self): 'name': 'SouthXchange', 'countries': 'AR', # Argentina 'rateLimit': 1000, - 'hasFetchTickers': True, - 'hasCORS': False, - 'hasWithdraw': True, + 'has': { + 'CORS': True, + 'fetchTickers': True, + 'withdraw': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27838912-4f94ec8a-60f6-11e7-9e5d-bbf9bd50a559.jpg', 'api': 'https://www.southxchange.com/api', diff --git a/python/ccxt/surbitcoin.py b/python/ccxt/surbitcoin.py index b7fdca0c9c91b..9714dc0abb09d 100644 --- a/python/ccxt/surbitcoin.py +++ b/python/ccxt/surbitcoin.py @@ -10,7 +10,9 @@ def describe(self): 'id': 'surbitcoin', 'name': 'SurBitcoin', 'countries': 'VE', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27991511-f0a50194-6481-11e7-99b5-8f02932424cc.jpg', 'api': { diff --git a/python/ccxt/therock.py b/python/ccxt/therock.py index 7f28c03b60db3..838575b782081 100644 --- a/python/ccxt/therock.py +++ b/python/ccxt/therock.py @@ -14,8 +14,10 @@ def describe(self): 'countries': 'MT', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': False, - 'hasFetchTickers': True, + 'has': { + 'CORS': False, + 'fetchTickers': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766869-75057fa2-5ee9-11e7-9a6f-13e641fa4707.jpg', 'api': 'https://api.therocktrading.com', diff --git a/python/ccxt/tidex.py b/python/ccxt/tidex.py index 20b2374a4d7db..ad2a9f9d50b92 100644 --- a/python/ccxt/tidex.py +++ b/python/ccxt/tidex.py @@ -12,8 +12,10 @@ def describe(self): 'countries': 'UK', 'rateLimit': 2000, 'version': '3', - # 'hasCORS': False, - # 'hasFetchTickers': True, + 'has': { + # 'CORS': False, + # 'fetchTickers': True + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30781780-03149dc4-a12e-11e7-82bb-313b269d24d4.jpg', 'api': { diff --git a/python/ccxt/urdubit.py b/python/ccxt/urdubit.py index 0049e0e7baf58..8319659fe459a 100644 --- a/python/ccxt/urdubit.py +++ b/python/ccxt/urdubit.py @@ -10,7 +10,9 @@ def describe(self): 'id': 'urdubit', 'name': 'UrduBit', 'countries': 'PK', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27991453-156bf3ae-6480-11e7-82eb-7295fe1b5bb4.jpg', 'api': { diff --git a/python/ccxt/vaultoro.py b/python/ccxt/vaultoro.py index 2229ae48f1889..c8fe2671a804c 100644 --- a/python/ccxt/vaultoro.py +++ b/python/ccxt/vaultoro.py @@ -12,7 +12,9 @@ def describe(self): 'countries': 'CH', 'rateLimit': 1000, 'version': '1', - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766880-f205e870-5ee9-11e7-8fe2-0d5b15880752.jpg', 'api': 'https://api.vaultoro.com', diff --git a/python/ccxt/vbtc.py b/python/ccxt/vbtc.py index 35657083e708d..53fff9c019da6 100644 --- a/python/ccxt/vbtc.py +++ b/python/ccxt/vbtc.py @@ -10,7 +10,9 @@ def describe(self): 'id': 'vbtc', 'name': 'VBTC', 'countries': 'VN', - 'hasCORS': False, + 'has': { + 'CORS': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27991481-1f53d1d8-6481-11e7-884e-21d17e7939db.jpg', 'api': { diff --git a/python/ccxt/virwox.py b/python/ccxt/virwox.py index 32a172b8eb6dc..0deea32edc18e 100644 --- a/python/ccxt/virwox.py +++ b/python/ccxt/virwox.py @@ -13,7 +13,9 @@ def describe(self): 'name': 'VirWoX', 'countries': ['AT', 'EU'], 'rateLimit': 1000, - 'hasCORS': True, + 'has': { + 'CORS': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766894-6da9d360-5eea-11e7-90aa-41f2711b7405.jpg', 'api': { diff --git a/python/ccxt/wex.py b/python/ccxt/wex.py index e9c81b8eaef9c..f8f32f9b31343 100644 --- a/python/ccxt/wex.py +++ b/python/ccxt/wex.py @@ -16,8 +16,10 @@ def describe(self): 'name': 'WEX', 'countries': 'NZ', # New Zealand 'version': '3', - 'hasFetchTickers': True, - 'hasCORS': False, + 'has': { + 'CORS': False, + 'fetchTickers': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/30652751-d74ec8f8-9e31-11e7-98c5-71469fcef03e.jpg', 'api': { diff --git a/python/ccxt/xbtce.py b/python/ccxt/xbtce.py index a63a0c37a7e0b..61e2d74f614af 100644 --- a/python/ccxt/xbtce.py +++ b/python/ccxt/xbtce.py @@ -16,10 +16,12 @@ def describe(self): 'countries': 'RU', 'rateLimit': 2000, # responses are cached every 2 seconds 'version': 'v1', - 'hasPublicAPI': False, - 'hasCORS': False, - 'hasFetchTickers': True, - 'hasFetchOHLCV': False, + 'has': { + 'publicAPI': False, + 'CORS': False, + 'fetchTickers': True, + 'fetchOHLCV': False, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/28059414-e235970c-662c-11e7-8c3a-08e31f78684b.jpg', 'api': 'https://cryptottlivewebapi.xbtce.net:8443/api', diff --git a/python/ccxt/yobit.py b/python/ccxt/yobit.py index d9ba9f469a8ee..2e449e027007a 100644 --- a/python/ccxt/yobit.py +++ b/python/ccxt/yobit.py @@ -15,8 +15,10 @@ def describe(self): 'countries': 'RU', 'rateLimit': 3000, # responses are cached every 2 seconds 'version': '3', - 'hasCORS': False, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'withdraw': True + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766910-cdcbfdae-5eea-11e7-9859-03fea873272d.jpg', 'api': { diff --git a/python/ccxt/yunbi.py b/python/ccxt/yunbi.py index 05b7eca6ef678..235c947fca920 100644 --- a/python/ccxt/yunbi.py +++ b/python/ccxt/yunbi.py @@ -12,9 +12,11 @@ def describe(self): 'countries': 'CN', 'rateLimit': 1000, 'version': 'v2', - 'hasCORS': False, - 'hasFetchTickers': True, - 'hasFetchOHLCV': True, + 'has': { + 'CORS': False, + 'fetchTickers': True, + 'fetchOHLCV': True, + }, 'timeframes': { '1m': '1', '5m': '5', diff --git a/python/ccxt/zaif.py b/python/ccxt/zaif.py index 8665fce95fbb9..0d4b1ed3cfefa 100644 --- a/python/ccxt/zaif.py +++ b/python/ccxt/zaif.py @@ -14,10 +14,12 @@ def describe(self): 'countries': 'JP', 'rateLimit': 2000, 'version': '1', - 'hasCORS': False, - 'hasFetchOpenOrders': True, - 'hasFetchClosedOrders': True, - 'hasWithdraw': True, + 'has': { + 'CORS': False, + 'fetchOpenOrders': True, + 'fetchClosedOrders': True, + 'withdraw': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/27766927-39ca2ada-5eeb-11e7-972f-1b4199518ca6.jpg', 'api': 'https://api.zaif.jp', diff --git a/python/ccxt/zb.py b/python/ccxt/zb.py index 180df7f148c0d..ac05a231c8795 100644 --- a/python/ccxt/zb.py +++ b/python/ccxt/zb.py @@ -15,8 +15,10 @@ def describe(self): 'countries': 'CN', 'rateLimit': 1000, 'version': 'v1', - 'hasCORS': False, - 'hasFetchOrder': True, + 'has': { + 'CORS': False, + 'fetchOrder': True, + }, 'urls': { 'logo': 'https://user-images.githubusercontent.com/1294454/32859187-cd5214f0-ca5e-11e7-967d-96568e2e2bd1.jpg', 'api': { diff --git a/python/test/test.py b/python/test/test.py index f0bce034d6856..332b95e5359ac 100644 --- a/python/test/test.py +++ b/python/test/test.py @@ -104,7 +104,7 @@ def handle_all_unhandled_exceptions(type, value, traceback): def test_order_book(exchange, symbol): - if exchange.hasFetchOrderBook: + if exchange.has['fetchOrderBook']: delay = int(exchange.rateLimit / 1000) time.sleep(delay) # dump(green(exchange.id), green(symbol), 'fetching order book...') @@ -125,7 +125,7 @@ def test_order_book(exchange, symbol): def test_ohlcv(exchange, symbol): - if exchange.hasFetchOHLCV: + if exchange.has['fetchOHLCV']: delay = int(exchange.rateLimit / 1000) time.sleep(delay) ohlcvs = exchange.fetch_ohlcv(symbol) @@ -137,7 +137,7 @@ def test_ohlcv(exchange, symbol): def test_tickers(exchange, symbol): - if exchange.hasFetchTickers: + if exchange.has['fetchTickers']: delay = int(exchange.rateLimit / 1000) time.sleep(delay) tickers = None @@ -165,7 +165,7 @@ def is_active_symbol(exchange, symbol): def test_ticker(exchange, symbol): - if exchange.hasFetchTicker: + if exchange.has['fetchTicker']: delay = int(exchange.rateLimit / 1000) time.sleep(delay) # dump(green(exchange.id), green(symbol), 'fetching ticker...') @@ -187,7 +187,7 @@ def test_ticker(exchange, symbol): def test_trades(exchange, symbol): - if exchange.hasFetchTrades: + if exchange.has['fetchTrades']: delay = int(exchange.rateLimit / 1000) time.sleep(delay) # dump(green(exchange.id), green(symbol), 'fetching trades...') @@ -266,7 +266,7 @@ def test_exchange(exchange): time.sleep(exchange.rateLimit / 1000) - if exchange.hasFetchOrders: + if exchange.has['fetchOrders']: try: # dump(green(exchange.id), 'fetching orders...') orders = exchange.fetch_orders(symbol) diff --git a/python/test/test_async.py b/python/test/test_async.py index 4a7b612758d02..bf86bc9d611ac 100644 --- a/python/test/test_async.py +++ b/python/test/test_async.py @@ -105,7 +105,7 @@ def handle_all_unhandled_exceptions(type, value, traceback): async def test_order_book(exchange, symbol): - if exchange.hasFetchOrderBook: + if exchange.has['fetchOrderBook']: delay = int(exchange.rateLimit / 1000) time.sleep(delay) # dump(green(exchange.id), green(symbol), 'fetching order book...') @@ -126,7 +126,7 @@ async def test_order_book(exchange, symbol): async def test_ohlcv(exchange, symbol): - if exchange.hasFetchOHLCV: + if exchange.has['fetchOHLCV']: delay = int(exchange.rateLimit / 1000) time.sleep(delay) ohlcvs = await exchange.fetch_ohlcv(symbol) @@ -138,7 +138,7 @@ async def test_ohlcv(exchange, symbol): async def test_tickers(exchange, symbol): - if exchange.hasFetchTickers: + if exchange.has['fetchTickers']: delay = int(exchange.rateLimit / 1000) time.sleep(delay) tickers = None @@ -194,7 +194,7 @@ async def test_l2_order_books_async(exchange): async def test_ticker(exchange, symbol): - if exchange.hasFetchTicker: + if exchange.has['fetchTicker']: delay = int(exchange.rateLimit / 1000) time.sleep(delay) # dump(green(exchange.id), green(symbol), 'fetching ticker...') @@ -216,7 +216,7 @@ async def test_ticker(exchange, symbol): async def test_trades(exchange, symbol): - if exchange.hasFetchTrades: + if exchange.has['fetchTrades']: delay = int(exchange.rateLimit / 1000) time.sleep(delay) # dump(green(exchange.id), green(symbol), 'fetching trades...') @@ -295,7 +295,7 @@ async def test_exchange(exchange): time.sleep(exchange.rateLimit / 1000) - if exchange.hasFetchOrders: + if exchange.has['fetchOrders']: try: # dump(green(exchange.id), 'fetching orders...') orders = await exchange.fetch_orders(symbol) diff --git a/run-tests.js b/run-tests.js index ff9ecb95fe2d6..a7f634781fd00 100644 --- a/run-tests.js +++ b/run-tests.js @@ -125,8 +125,6 @@ const sequentialMap = async (input, fn) => { const testExchange = async (exchange) => { - const nonce = ccxt.time.now () - /* Run tests for all/selected languages (in parallel) */ const args = [exchange, ...symbol === 'all' ? [] : symbol] diff --git a/wiki/Manual.md b/wiki/Manual.md index b35698c8418fc..654a7589ff28d 100644 --- a/wiki/Manual.md +++ b/wiki/Manual.md @@ -236,9 +236,32 @@ Here's an overview of base exchange properties with values added for example: }, 'version': 'v1', // string ending with digits 'api': { ... }, // dictionary of api endpoints - 'hasFetchTickers': true, // true if the exchange implements fetchTickers () - 'hasFetchOHLCV': false, // true if the exchange implements fetchOHLCV () - 'timeframes': { // empty if the exchange !hasFetchOHLCV + 'has': { // exchange capabilities + 'CORS': false, + 'publicAPI': true, + 'privateAPI': true, + 'cancelOrder': true, + 'createDepositAddress': false, + 'createOrder': true, + 'deposit': false, + 'fetchBalance': true, + 'fetchClosedOrders': false, + 'fetchCurrencies': false, + 'fetchDepositAddress': false, + 'fetchMarkets': true, + 'fetchMyTrades': false, + 'fetchOHLCV': false, + 'fetchOpenOrders': false, + 'fetchOrder': false, + 'fetchOrderBook': true, + 'fetchOrders': false, + 'fetchTicker': true, + 'fetchTickers': false, + 'fetchBidsAsks': false, + 'fetchTrades': true, + 'withdraw': false, + }, + 'timeframes': { // empty if the exchange !has.fetchOHLCV '1m': '1minute', '1h': '1hour', '1d': '1day', @@ -281,9 +304,7 @@ Below is a detailed description of each of the base exchange properties: - `api`: An associative array containing a definition of all API endpoints exposed by a crypto exchange. The API definition is used by ccxt to automatically construct callable instance methods for each available endpoint. -- `hasFetchTickers`: This is a boolean property indicating if the exchange has the fetchTickers () method available. When this property is false, the exchange will also throw a NotSupported exception upon a call to fetchTickers (). - -- `hasFetchOHLCV`: This is a boolean property indicating if the exchange has the fetchOHLCV () method available. When this property is false, the exchange will also throw a NotSupported exception upon a call to fetchOHLCV (). Also, if this property is true, the `timeframes` property is populated as well. +- `has`: This is an associative array of exchange capabilities (e.g `fetchTickers`, `fetchOHLCV` or `CORS`). - `timeframes`: An associative array of timeframes, supported by the fetchOHLCV method of the exchange. This is only populated when `hasFetchTickers` property is true. @@ -1019,7 +1040,7 @@ You can call the unified `fetchOHLCV` / `fetch_ohlcv` method to get the list of ```JavaScript // JavaScript let sleep = (ms) => new Promise (resolve => setTimeout (resolve, ms)); -if (exchange.hasFetchOHLCV) { +if (exchange.has.fetchOHLCV) { (async () => { for (symbol in exchange.markets) { await sleep (exchange.rateLimit) // milliseconds